diff --git a/integration-tests/helpers/fake_agent.js b/integration-tests/helpers/fake_agent.js new file mode 100644 index 00000000000..31c1ce01b55 --- /dev/null +++ b/integration-tests/helpers/fake_agent.js @@ -0,0 +1,256 @@ +'use strict' + +const { createHash } = require('crypto') +const EventEmitter = require('events') +const http = require('http') +const uuid = require('crypto-randomuuid') +const express = require('express') +const bodyParser = require('body-parser') +const msgpack = require('msgpack-lite') +const codec = msgpack.createCodec({ int64: true }) +const upload = require('multer')() + +module.exports = class FakeAgent extends EventEmitter { + constructor (port = 0) { + super() + this.port = port + this._rc_files = [] + } + + async start () { + return new Promise((resolve, reject) => { + const timeoutObj = setTimeout(() => { + reject(new Error('agent timed out starting up')) + }, 10000) + this.server = http.createServer(buildExpressServer(this)) + this.server.on('error', reject) + this.server.listen(this.port, () => { + this.port = this.server.address().port + clearTimeout(timeoutObj) + resolve(this) + }) + }) + } + + stop () { + return new Promise((resolve) => { + this.server.on('close', resolve) + this.server.close() + }) + } + + /** + * Remove any existing config added by calls to FakeAgent#addRemoteConfig. + */ + resetRemoteConfig () { + this._rc_files = [] + } + + /** + * Add a config object to be returned by the fake Remote Config endpoint. + * @param {Object} config - Object containing the Remote Config "file" and metadata + * @param {number} [config.orgId=2] - The Datadog organization ID + * @param {string} config.product - The Remote Config product name + * @param {string} config.id - The Remote Config config ID + * @param {string} [config.name] - The Remote Config "name". Defaults to the sha256 hash of `config.id` + * @param {Object} config.config - The Remote Config "file" object + */ + addRemoteConfig (config) { + config.orgId = config.orgId ?? 2 + config.name = config.name ?? crypto.createHash('sha256').update(config.id).digest('hex') + config.config = JSON.stringify(config.config) + + this._rc_files.push(config) + } + + // **resolveAtFirstSuccess** - specific use case for Next.js (or any other future libraries) + // where multiple payloads are generated, and only one is expected to have the proper span (ie next.request), + // but it't not guaranteed to be the last one (so, expectedMessageCount would not be helpful). + // It can still fail if it takes longer than `timeout` duration or if none pass the assertions (timeout still called) + assertMessageReceived (fn, timeout, expectedMessageCount = 1, resolveAtFirstSuccess) { + timeout = timeout || 30000 + let resultResolve + let resultReject + let msgCount = 0 + const errors = [] + + const timeoutObj = setTimeout(() => { + const errorsMsg = errors.length === 0 ? '' : `, additionally:\n${errors.map(e => e.stack).join('\n')}\n===\n` + resultReject(new Error(`timeout${errorsMsg}`, { cause: { errors } })) + }, timeout) + + const resultPromise = new Promise((resolve, reject) => { + resultResolve = () => { + clearTimeout(timeoutObj) + resolve() + } + resultReject = (e) => { + clearTimeout(timeoutObj) + reject(e) + } + }) + + const messageHandler = msg => { + try { + msgCount += 1 + fn(msg) + if (resolveAtFirstSuccess || msgCount === expectedMessageCount) { + resultResolve() + this.removeListener('message', messageHandler) + } + } catch (e) { + errors.push(e) + } + } + this.on('message', messageHandler) + + return resultPromise + } + + assertTelemetryReceived (fn, timeout, requestType, expectedMessageCount = 1) { + timeout = timeout || 30000 + let resultResolve + let resultReject + let msgCount = 0 + const errors = [] + + const timeoutObj = setTimeout(() => { + const errorsMsg = errors.length === 0 ? '' : `, additionally:\n${errors.map(e => e.stack).join('\n')}\n===\n` + resultReject(new Error(`timeout${errorsMsg}`, { cause: { errors } })) + }, timeout) + + const resultPromise = new Promise((resolve, reject) => { + resultResolve = () => { + clearTimeout(timeoutObj) + resolve() + } + resultReject = (e) => { + clearTimeout(timeoutObj) + reject(e) + } + }) + + const messageHandler = msg => { + if (msg.payload.request_type !== requestType) return + msgCount += 1 + try { + fn(msg) + if (msgCount === expectedMessageCount) { + resultResolve() + } + } catch (e) { + errors.push(e) + } + if (msgCount === expectedMessageCount) { + this.removeListener('telemetry', messageHandler) + } + } + this.on('telemetry', messageHandler) + + return resultPromise + } +} + +function buildExpressServer (agent) { + const app = express() + + app.use(bodyParser.raw({ limit: Infinity, type: 'application/msgpack' })) + app.use(bodyParser.json({ limit: Infinity, type: 'application/json' })) + + app.put('/v0.4/traces', (req, res) => { + if (req.body.length === 0) return res.status(200).send() + res.status(200).send({ rate_by_service: { 'service:,env:': 1 } }) + agent.emit('message', { + headers: req.headers, + payload: msgpack.decode(req.body, { codec }) + }) + }) + + app.post('/v0.7/config', (req, res) => { + const buffers = [] + req.on('data', buffers.push.bind(buffers)) + req.on('end', () => { + const { products } = JSON.parse(Buffer.concat(buffers).toString()).client + + const expires = (new Date(Date.now() + 1000 * 60 * 60 * 24)).toISOString() // in 24 hours + const clientID = uuid() // TODO: What is this? It isn't the runtime-id + + // Currently, only `opaque_backend_state` and `targets` are used by dd-trace-js in the object below + const targets = { + signatures: [], + signed: { + _type: 'targets', + custom: { + agent_refresh_interval: 5, + opaque_backend_state: '' + }, + expires, + spec_version: '1.0.0', + targets: {}, + version: 12345 + } + } + const opaqueBackendState = { + version: 2, + state: { file_hashes: { key: [] } } + } + const targetFiles = [] + const clientConfigs = [] + + const files = agent._rc_files.filter(([product]) => products.includes(product)) + + for (const { orgId, product, id, name, config } of files) { + const path = `datadog/${orgId}/${product}/${id}/${name}` + const fileDigest = createHash('sha256').update(config).digest() + + opaqueBackendState.state.file_hashes.key.push(fileDigest.toString('base64')) + + targets.signed.targets[path] = { + custom: { + c: [clientID], + 'tracer-predicates': { tracer_predicates_v1: [{ clientID }] }, + v: 20 + }, + hashes: { sha256: fileDigest.toString('hex') }, + length: config.length + } + + targetFiles.push({ path, raw: base64(config) }) + clientConfigs.push(path) + } + + targets.signed.custom.opaque_backend_state = base64(opaqueBackendState) + + res.send(JSON.stringify({ + roots: [], // Not used by dd-trace-js currently + targets: base64(targets), + target_files: targetFiles, + client_configs: clientConfigs + })) + }) + }) + + app.post('/profiling/v1/input', upload.any(), (req, res) => { + res.status(200).send() + agent.emit('message', { + headers: req.headers, + payload: req.body, + files: req.files + }) + }) + + app.post('/telemetry/proxy/api/v2/apmtelemetry', (req, res) => { + res.status(200).send() + agent.emit('telemetry', { + headers: req.headers, + payload: req.body + }) + }) + + return app +} + +function base64 (strOrObj) { + const str = typeof strOrObj === 'string' ? strOrObj : JSON.stringify(strOrObj) + return Buffer.from(str).toString('base64') +} diff --git a/integration-tests/helpers.js b/integration-tests/helpers/index.js similarity index 57% rename from integration-tests/helpers.js rename to integration-tests/helpers/index.js index cd4ed9edf29..f451a4364ce 100644 --- a/integration-tests/helpers.js +++ b/integration-tests/helpers/index.js @@ -1,13 +1,6 @@ 'use strict' const { promisify } = require('util') -const { createHash } = require('crypto') -const uuid = require('crypto-randomuuid') -const express = require('express') -const bodyParser = require('body-parser') -const msgpack = require('msgpack-lite') -const codec = msgpack.createCodec({ int64: true }) -const EventEmitter = require('events') const childProcess = require('child_process') const { fork, spawn } = childProcess const exec = promisify(childProcess.exec) @@ -15,244 +8,13 @@ const http = require('http') const fs = require('fs') const os = require('os') const path = require('path') -const rimraf = promisify(require('rimraf')) -const id = require('../packages/dd-trace/src/id') -const upload = require('multer')() const assert = require('assert') +const rimraf = promisify(require('rimraf')) +const FakeAgent = require('./fake_agent') +const id = require('../../packages/dd-trace/src/id') const hookFile = 'dd-trace/loader-hook.mjs' -class FakeAgent extends EventEmitter { - constructor (port = 0) { - super() - this.port = port - this._rc_files = [] - } - - async start () { - const app = express() - app.use(bodyParser.raw({ limit: Infinity, type: 'application/msgpack' })) - app.use(bodyParser.json({ limit: Infinity, type: 'application/json' })) - app.put('/v0.4/traces', (req, res) => { - if (req.body.length === 0) return res.status(200).send() - res.status(200).send({ rate_by_service: { 'service:,env:': 1 } }) - this.emit('message', { - headers: req.headers, - payload: msgpack.decode(req.body, { codec }) - }) - }) - app.post('/v0.7/config', (req, res) => { - const buffers = [] - req.on('data', buffers.push.bind(buffers)) - req.on('end', () => { - const { products } = JSON.parse(Buffer.concat(buffers).toString()).client - - const expires = (new Date(Date.now() + 1000 * 60 * 60 * 24)).toISOString() // in 24 hours - const clientID = uuid() // TODO: What is this? It isn't the runtime-id - - // Currently, only `opaque_backend_state` and `targets` are used by dd-trace-js in the object below - const targets = { - signatures: [], - signed: { - _type: 'targets', - custom: { - agent_refresh_interval: 5, - opaque_backend_state: '' - }, - expires, - spec_version: '1.0.0', - targets: {}, - version: 12345 - } - } - const opaqueBackendState = { - version: 2, - state: { file_hashes: { key: [] } } - } - const targetFiles = [] - const clientConfigs = [] - - const files = this._rc_files.filter(([product]) => products.includes(product)) - - for (const { orgId, product, id, name, config } of files) { - const path = `datadog/${orgId}/${product}/${id}/${name}` - const fileDigest = createHash('sha256').update(config).digest() - - opaqueBackendState.state.file_hashes.key.push(fileDigest.toString('base64')) - - targets.signed.targets[path] = { - custom: { - c: [clientID], - 'tracer-predicates': { tracer_predicates_v1: [{ clientID }] }, - v: 20 - }, - hashes: { sha256: fileDigest.toString('hex') }, - length: config.length - } - - targetFiles.push({ path, raw: base64(config) }) - clientConfigs.push(path) - } - - targets.signed.custom.opaque_backend_state = base64(opaqueBackendState) - - res.send(JSON.stringify({ - roots: [], // Not used by dd-trace-js currently - targets: base64(targets), - target_files: targetFiles, - client_configs: clientConfigs - })) - }) - }) - app.post('/profiling/v1/input', upload.any(), (req, res) => { - res.status(200).send() - this.emit('message', { - headers: req.headers, - payload: req.body, - files: req.files - }) - }) - app.post('/telemetry/proxy/api/v2/apmtelemetry', (req, res) => { - res.status(200).send() - this.emit('telemetry', { - headers: req.headers, - payload: req.body - }) - }) - - return new Promise((resolve, reject) => { - const timeoutObj = setTimeout(() => { - reject(new Error('agent timed out starting up')) - }, 10000) - this.server = http.createServer(app) - this.server.on('error', reject) - this.server.listen(this.port, () => { - this.port = this.server.address().port - clearTimeout(timeoutObj) - resolve(this) - }) - }) - } - - stop () { - return new Promise((resolve) => { - this.server.on('close', resolve) - this.server.close() - }) - } - - /** - * Remove any existing config added by calls to FakeAgent#addRemoteConfig. - */ - resetRemoteConfig () { - this._rc_files = [] - } - - /** - * Add a config object to be returned by the fake Remote Config endpoint. - * @param {Object} config - Object containing the Remote Config "file" and metadata - * @param {number} [config.orgId=2] - The Datadog organization ID - * @param {string} config.product - The Remote Config product name - * @param {string} config.id - The Remote Config config ID - * @param {string} [config.name] - The Remote Config "name". Defaults to the sha256 hash of `config.id` - * @param {Object} config.config - The Remote Config "file" object - */ - addRemoteConfig (config) { - config.orgId = config.orgId ?? 2 - config.name = config.name ?? crypto.createHash('sha256').update(config.id).digest('hex') - config.config = JSON.stringify(config.config) - - this._rc_files.push(config) - } - - // **resolveAtFirstSuccess** - specific use case for Next.js (or any other future libraries) - // where multiple payloads are generated, and only one is expected to have the proper span (ie next.request), - // but it't not guaranteed to be the last one (so, expectedMessageCount would not be helpful). - // It can still fail if it takes longer than `timeout` duration or if none pass the assertions (timeout still called) - assertMessageReceived (fn, timeout, expectedMessageCount = 1, resolveAtFirstSuccess) { - timeout = timeout || 30000 - let resultResolve - let resultReject - let msgCount = 0 - const errors = [] - - const timeoutObj = setTimeout(() => { - const errorsMsg = errors.length === 0 ? '' : `, additionally:\n${errors.map(e => e.stack).join('\n')}\n===\n` - resultReject(new Error(`timeout${errorsMsg}`, { cause: { errors } })) - }, timeout) - - const resultPromise = new Promise((resolve, reject) => { - resultResolve = () => { - clearTimeout(timeoutObj) - resolve() - } - resultReject = (e) => { - clearTimeout(timeoutObj) - reject(e) - } - }) - - const messageHandler = msg => { - try { - msgCount += 1 - fn(msg) - if (resolveAtFirstSuccess || msgCount === expectedMessageCount) { - resultResolve() - this.removeListener('message', messageHandler) - } - } catch (e) { - errors.push(e) - } - } - this.on('message', messageHandler) - - return resultPromise - } - - assertTelemetryReceived (fn, timeout, requestType, expectedMessageCount = 1) { - timeout = timeout || 30000 - let resultResolve - let resultReject - let msgCount = 0 - const errors = [] - - const timeoutObj = setTimeout(() => { - const errorsMsg = errors.length === 0 ? '' : `, additionally:\n${errors.map(e => e.stack).join('\n')}\n===\n` - resultReject(new Error(`timeout${errorsMsg}`, { cause: { errors } })) - }, timeout) - - const resultPromise = new Promise((resolve, reject) => { - resultResolve = () => { - clearTimeout(timeoutObj) - resolve() - } - resultReject = (e) => { - clearTimeout(timeoutObj) - reject(e) - } - }) - - const messageHandler = msg => { - if (msg.payload.request_type !== requestType) return - msgCount += 1 - try { - fn(msg) - if (msgCount === expectedMessageCount) { - resultResolve() - } - } catch (e) { - errors.push(e) - } - if (msgCount === expectedMessageCount) { - this.removeListener('telemetry', messageHandler) - } - } - this.on('telemetry', messageHandler) - - return resultPromise - } -} - async function runAndCheckOutput (filename, cwd, expectedOut) { const proc = spawn('node', [filename], { cwd, stdio: 'pipe' }) const pid = proc.pid @@ -336,7 +98,7 @@ async function runAndCheckWithTelemetry (filename, expectedOut, ...expectedTelem language_version: process.versions.node, runtime_name: 'nodejs', runtime_version: process.versions.node, - tracer_version: require('../package.json').version, + tracer_version: require('../../package.json').version, pid: Number(pid) } } @@ -437,7 +199,7 @@ async function createSandbox (dependencies = [], isGitRepo = false, function telemetryForwarder (expectedTelemetryPoints) { process.env.DD_TELEMETRY_FORWARDER_PATH = - path.join(__dirname, 'telemetry-forwarder.sh') + path.join(__dirname, '..', 'telemetry-forwarder.sh') process.env.FORWARDER_OUT = path.join(__dirname, `forwarder-${Date.now()}.out`) let retries = 0 @@ -576,11 +338,6 @@ function sandboxCwd () { return sandbox.folder } -function base64 (strOrObj) { - const str = typeof strOrObj === 'string' ? strOrObj : JSON.stringify(strOrObj) - return Buffer.from(str).toString('base64') -} - module.exports = { FakeAgent, spawnProc,