-
Notifications
You must be signed in to change notification settings - Fork 312
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add support for Dynamic Instrumentation
- Loading branch information
Showing
15 changed files
with
954 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
'use strict' | ||
|
||
const { workerData: { config: parentConfig, configPort } } = require('node:worker_threads') | ||
const { URL, format } = require('node:url') | ||
|
||
const config = module.exports = { | ||
runtimeId: parentConfig.tags['runtime-id'], | ||
service: parentConfig.service | ||
} | ||
|
||
updateUrl(parentConfig) | ||
|
||
configPort.on('message', updateUrl) | ||
|
||
function updateUrl (updates) { | ||
// if (!updates.url && !updates.hostname && !updates.port) return | ||
|
||
const url = updates.url || new URL(format({ | ||
// TODO: Can this ever be anything other than `http:`, and if so, how do we get the configured value? | ||
protocol: config.url?.protocol || 'http:', | ||
hostname: updates.hostname || config.url?.hostname || 'localhost', | ||
port: updates.port || config.url?.port | ||
})) | ||
|
||
config.url = url.toString() | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
'use strict' | ||
|
||
const uuid = require('crypto-randomuuid') | ||
const { breakpoints } = require('./state') | ||
const session = require('./session') | ||
const { getLocalStateForBreakpoint } = require('./snapshot') | ||
const send = require('./send') | ||
const { ackEmitting, ackError } = require('./status') | ||
require('./remote_config') | ||
const log = require('../../log') | ||
|
||
// The `session.connectToMainThread()` method called above doesn't "register" any active handles, so the worker thread | ||
// will exit with code 0 once when reaches the end of the file unless we do something to keep it alive: | ||
setInterval(() => {}, 1000 * 60) | ||
|
||
session.on('Debugger.paused', async ({ params }) => { | ||
const start = process.hrtime.bigint() | ||
const timestamp = Date.now() | ||
|
||
let captureSnapshotForProbe = null | ||
const probes = params.hitBreakpoints.map((id) => { | ||
const probe = breakpoints.get(id) | ||
if (captureSnapshotForProbe === null && probe.captureSnapshot) captureSnapshotForProbe = probe | ||
return probe | ||
}) | ||
|
||
let state | ||
if (captureSnapshotForProbe !== null) { | ||
try { | ||
state = await getLocalStateForBreakpoint(params) | ||
} catch (err) { | ||
ackError(err, captureSnapshotForProbe) // TODO: Ok to continue after sending ackError? | ||
} | ||
} | ||
|
||
await session.post('Debugger.resume') | ||
const diff = process.hrtime.bigint() - start // TODO: Should we log this using some sort of telemetry? | ||
|
||
log.debug(`Finished processing breakpoints - thread paused for: ${Number(diff) / 1000000} ms`) | ||
|
||
// TODO: Is this the correct way of handling multiple breakpoints hit at the same time? | ||
for (const probe of probes) { | ||
const captures = probe.captureSnapshot && state | ||
? { | ||
lines: { | ||
[probe.location.lines[0]]: state | ||
} | ||
} | ||
: undefined | ||
|
||
await send({ | ||
message: probe.template, // TODO: Process template | ||
snapshot: { | ||
id: uuid(), | ||
timestamp, | ||
captures, | ||
probe: { | ||
id: probe.id, | ||
version: probe.version, // TODO: Should this always be 2??? | ||
location: probe.location | ||
}, | ||
language: 'javascript' | ||
} | ||
}) | ||
|
||
ackEmitting(probe) | ||
|
||
// TODO: Remove before shipping | ||
process._rawDebug( | ||
'\nLocal state:\n' + | ||
'--------------------------------------------------\n' + | ||
stateToString(state) + | ||
'--------------------------------------------------\n' + | ||
'\nStats:\n' + | ||
'--------------------------------------------------\n' + | ||
` Total state JSON size: ${JSON.stringify(state).length} bytes\n` + | ||
`Processed was paused for: ${Number(diff) / 1000000} ms\n` + | ||
'--------------------------------------------------\n' | ||
) | ||
} | ||
}) | ||
|
||
// TODO: Remove this function before shipping | ||
function stateToString (state) { | ||
if (state === undefined) return '<not captured>' | ||
let str = '' | ||
for (const [name, value] of Object.entries(state)) { | ||
str += `${name}: ${color(value)}\n` | ||
} | ||
return str | ||
} | ||
|
||
// TODO: Remove this function before shipping | ||
function color (obj) { | ||
return require('node:util').inspect(obj, { depth: null, colors: true }) | ||
} |
23 changes: 23 additions & 0 deletions
23
packages/dd-trace/src/debugger/devtools_client/inspector_promises_polyfill.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
'use strict' | ||
|
||
const { builtinModules } = require('node:module') | ||
|
||
if (builtinModules.includes('inspector/promises')) { | ||
module.exports = require('node:inspector/promises') | ||
} else { | ||
const inspector = require('node:inspector') | ||
const { promisify } = require('node:util') | ||
|
||
// The rest of the code in this file is lifted from: | ||
// https://github.com/nodejs/node/blob/1d4d76ff3fb08f9a0c55a1d5530b46c4d5d550c7/lib/inspector/promises.js | ||
class Session extends inspector.Session { | ||
constructor () { super() } // eslint-disable-line no-useless-constructor | ||
} | ||
|
||
Session.prototype.post = promisify(inspector.Session.prototype.post) | ||
|
||
module.exports = { | ||
...inspector, | ||
Session | ||
} | ||
} |
119 changes: 119 additions & 0 deletions
119
packages/dd-trace/src/debugger/devtools_client/remote_config.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
'use strict' | ||
|
||
const { workerData: { rcPort } } = require('node:worker_threads') | ||
const { scripts, probes, breakpoints } = require('./state') | ||
const session = require('./session') | ||
const { ackReceived, ackInstalled, ackError } = require('./status') | ||
const log = require('../../log') | ||
|
||
let sessionStarted = false | ||
|
||
// Example `probe`: | ||
// { | ||
// id: '100c9a5c-45ad-49dc-818b-c570d31e11d1', | ||
// version: 0, | ||
// type: 'LOG_PROBE', | ||
// language: 'javascript', // ignore | ||
// where: { | ||
// sourceFile: 'index.js', | ||
// lines: ['25'] // only use first array element | ||
// }, | ||
// tags: [], // not used | ||
// template: 'Hello World 2', | ||
// segments: [{ str: 'Hello World 2' }], | ||
// captureSnapshot: true, | ||
// capture: { maxReferenceDepth: 1 }, | ||
// sampling: { snapshotsPerSecond: 1 }, | ||
// evaluateAt: 'EXIT' // only used for method probes | ||
// } | ||
rcPort.on('message', async ({ action, conf: probe }) => { | ||
try { | ||
await processMsg(action, probe) | ||
} catch (err) { | ||
ackError(err, probe) | ||
} | ||
}) | ||
|
||
async function start () { | ||
sessionStarted = true | ||
await session.post('Debugger.enable') | ||
} | ||
|
||
async function stop () { | ||
sessionStarted = false | ||
await session.post('Debugger.disable') | ||
} | ||
|
||
async function processMsg (action, probe) { | ||
log.debug(`Received request to ${action} ${probe.type} probe (id: ${probe.id}, version: ${probe.version})`) | ||
|
||
ackReceived(probe) | ||
|
||
if (probe.type !== 'LOG_PROBE') { | ||
throw new Error(`Unsupported probe type: ${probe.type} (id: ${probe.id}, version: ${probe.version})`) | ||
// TODO: Throw also if method log probe | ||
} | ||
|
||
switch (action) { | ||
case 'unapply': | ||
await removeBreakpoint(probe) | ||
break | ||
case 'apply': | ||
await addBreakpoint(probe) | ||
break | ||
case 'modify': | ||
// TODO: Can we modify in place? | ||
await removeBreakpoint(probe) | ||
await addBreakpoint(probe) | ||
break | ||
default: | ||
throw new Error( | ||
`Cannot process probe ${probe.id} (version: ${probe.version}) - unknown remote configuration action: ${action}` | ||
) | ||
} | ||
} | ||
|
||
async function addBreakpoint (probe) { | ||
if (!sessionStarted) await start() | ||
|
||
// Optimize for sending data to /debugger/v1/input endpoint | ||
probe.location = { | ||
file: probe.where.sourceFile, | ||
lines: [Number(probe.where.lines[0])] // Tracer doesn't support multiple-line breakpoints | ||
} | ||
delete probe.where | ||
|
||
// TODO: Figure out what to do about the full path | ||
const path = `file:///Users/thomas.watson/go/src/github.com/DataDog/debugger-demos/nodejs/${probe.location.file}` | ||
|
||
const { breakpointId } = await session.post('Debugger.setBreakpoint', { | ||
location: { | ||
scriptId: scripts.get(path), | ||
lineNumber: probe.location.lines[0] - 1 // Beware! lineNumber is zero-indexed | ||
} | ||
// TODO: Support conditions | ||
// condition: "request.params.name === 'break'" | ||
}) | ||
|
||
probes.set(probe.id, breakpointId) | ||
breakpoints.set(breakpointId, probe) | ||
|
||
ackInstalled(probe) | ||
} | ||
|
||
async function removeBreakpoint ({ id }) { | ||
if (!sessionStarted) { | ||
// We should not get in this state, but abort if we do so the code doesn't throw | ||
throw Error(`Cannot remove probe ${id}: Debugger not started`) | ||
} | ||
if (!probes.has(id)) { | ||
throw Error(`Unknown probe id: ${id}`) | ||
} | ||
|
||
const breakpointId = probes.get(id) | ||
await session.post('Debugger.removeBreakpoint', { breakpointId }) | ||
probes.delete(id) | ||
breakpoints.delete(breakpointId) | ||
|
||
if (breakpoints.size === 0) await stop() | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
'use strict' | ||
|
||
const { hostname } = require('node:os') | ||
const { inspect } = require('node:util') | ||
const config = require('./config') | ||
const request = require('../../exporters/common/request') | ||
|
||
const host = hostname() | ||
const service = config.service | ||
const ddsource = 'dd_debugger' | ||
|
||
// TODO: Figure out correct logger values | ||
const logger = { | ||
name: __filename, | ||
method: '<module>', | ||
thread_name: `${process.argv0};pid:${process.pid}`, | ||
thread_id: 42, | ||
version: 2 | ||
} | ||
|
||
module.exports = async function send ({ message, snapshot: { id, timestamp, captures, probe, language } }) { | ||
const opts = { | ||
method: 'POST', | ||
url: config.url, | ||
path: '/debugger/v1/input', | ||
headers: { | ||
'Content-Type': 'application/json; charset=utf-8' | ||
// Accept: 'text/plain' // TODO: This seems wrong (from Python tracer) | ||
} | ||
} | ||
|
||
const payload = { | ||
service, | ||
ddsource, | ||
message, | ||
logger, | ||
// 'dd.trace_id': null, | ||
// 'dd.span_id': null, | ||
// method, | ||
'debugger.snapshot': { | ||
id, | ||
timestamp, | ||
captures, | ||
// evaluationErrors: [{ expr: 'foo == 42', message: 'foo' }], // TODO: This doesn't seem to be documented? | ||
probe, | ||
language | ||
}, | ||
host, // TODO: This doesn't seem to be documented? | ||
timestamp: Date.now() // TODO: This doesn't seem to be documented? | ||
// ddtags: {} // TODO: This doesn't seem to be documented? | ||
} | ||
|
||
process._rawDebug('Payload:', inspect(payload, { depth: null, colors: true })) // TODO: Remove | ||
|
||
request(JSON.stringify(payload), opts, (err, data, statusCode) => { | ||
if (err) throw err // TODO: Handle error | ||
process._rawDebug('Response:', { statusCode }) // TODO: Remove | ||
process._rawDebug('Response body:', JSON.parse(data)) // TODO: Remove | ||
}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
'use strict' | ||
|
||
const inspector = require('./inspector_promises_polyfill') | ||
|
||
const session = module.exports = new inspector.Session() | ||
|
||
session.connectToMainThread() |
Oops, something went wrong.