Skip to content

Commit

Permalink
[test] allow FakeAgent to respond to remote config requests (#4626)
Browse files Browse the repository at this point in the history
Allow integration tests to mock the remote config abilities of the agent.

This is done by first setting up a new remote config "file" using the new
`FakeAgent#addRemoteConfig` method before running the main part of the
integration test. This primes the fake agent to respond with this config for
http requests to the `/v0.7/config` endpoint.
  • Loading branch information
watson authored and bengl committed Aug 30, 2024
1 parent f667ec0 commit dbc7e2b
Show file tree
Hide file tree
Showing 4 changed files with 322 additions and 189 deletions.
298 changes: 298 additions & 0 deletions integration-tests/helpers/fake-agent.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,298 @@
'use strict'

const { createHash } = require('crypto')
const EventEmitter = require('events')
const http = require('http')
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._rcFiles = {}
this._rcTargetsVersion = 0
}

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()
})
}

/**
* 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 = { ...config }
config.orgId = config.orgId || 2
config.name = config.name || createHash('sha256').update(config.id).digest('hex')
config.config = JSON.stringify(config.config)
config.path = `datadog/${config.orgId}/${config.product}/${config.id}/${config.name}`
config.fileHash = createHash('sha256').update(config.config).digest('hex')
config.meta = {
custom: { v: 1 },
hashes: { sha256: config.fileHash },
length: config.config.length
}

this._rcFiles[config.id] = config
this._rcTargetsVersion++
}

/**
* Update an existing config object
* @param {string} id - The Remote Config config ID
* @param {Object} config - The Remote Config "file" object
*/
updateRemoteConfig (id, config) {
config = JSON.stringify(config)
config = Object.assign(
this._rcFiles[id],
{
config,
fileHash: createHash('sha256').update(config).digest('hex')
}
)
config.meta.custom.v++
config.meta.hashes.sha256 = config.fileHash
config.meta.length = config.config.length
this._rcTargetsVersion++
}

/**
* Remove a specific config object
* @param {string} id - The ID of the config object that should be removed
*/
removeRemoteConfig (id) {
delete this._rcFiles[id]
this._rcTargetsVersion++
}

/**
* Remove any existing config added by calls to FakeAgent#addRemoteConfig.
*/
resetRemoteConfig () {
this._rcFiles = {}
this._rcTargetsVersion++
}

// **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 {
client: { products, state },
cached_target_files: cachedTargetFiles
} = req.body

if (state.has_error) {
// Print the error sent by the client in case it's useful in debugging tests
console.error(state.error) // eslint-disable-line no-console
}

for (const { apply_error: error } of state.config_states) {
if (error) {
// Print the error sent by the client in case it's useful in debugging tests
console.error(error) // eslint-disable-line no-console
}
}

if (agent._rcTargetsVersion === state.targets_version) {
// If the state hasn't changed since the last time the client asked, just return an empty result
res.json({})
return
}

if (Object.keys(agent._rcFiles).length === 0) {
// All config files have been removed, but the client has not yet been informed.
// Return this custom result to let the client know.
res.json({ client_configs: [] })
return
}

// The actual targets object is much more complicated,
// but the Node.js tracer currently only cares about the following properties.
const targets = {
signed: {
custom: { opaque_backend_state: 'foo' },
targets: {},
version: agent._rcTargetsVersion
}
}
const targetFiles = []
const clientConfigs = []

const files = Object.values(agent._rcFiles).filter(({ product }) => products.includes(product))

for (const { path, fileHash, meta, config } of files) {
clientConfigs.push(path)
targets.signed.targets[path] = meta

// skip files already cached by the client so we don't send them more than once
if (cachedTargetFiles.some((cached) =>
path === cached.path &&
fileHash === cached.hashes.find((e) => e.algorithm === 'sha256').hash
)) continue

targetFiles.push({ path, raw: base64(config) })
}

// The real response object also contains a `roots` property which has been omitted here since it's not currently
// used by the Node.js tracer.
res.json({
targets: clientConfigs.length === 0 ? undefined : 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')
}
Loading

0 comments on commit dbc7e2b

Please sign in to comment.