-
Notifications
You must be signed in to change notification settings - Fork 309
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
12 changed files
with
478 additions
and
5 deletions.
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,24 @@ | ||
'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) { | ||
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,38 @@ | ||
'use strict' | ||
|
||
const uuid = require('crypto-randomuuid') | ||
const { breakpoints } = require('./state') | ||
const session = require('./session') | ||
const send = require('./send') | ||
const { ackEmitting } = require('./status') | ||
require('./remote_config') | ||
const log = require('../../log') | ||
|
||
session.on('Debugger.paused', async ({ params }) => { | ||
const start = process.hrtime.bigint() | ||
const timestamp = Date.now() | ||
const probes = params.hitBreakpoints.map((id) => breakpoints.get(id)) | ||
await session.post('Debugger.resume') | ||
const diff = process.hrtime.bigint() - start // TODO: Should this be recored as telemetry? | ||
|
||
log.debug(`Finished processing breakpoints - main 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) { | ||
await send( | ||
probe.template, // TODO: Process template | ||
{ | ||
id: uuid(), | ||
timestamp, | ||
probe: { | ||
id: probe.id, | ||
version: probe.version, | ||
location: probe.location | ||
}, | ||
language: 'javascript' | ||
} | ||
) | ||
|
||
ackEmitting(probe) | ||
} | ||
}) |
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 | ||
} | ||
} |
134 changes: 134 additions & 0 deletions
134
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,134 @@ | ||
'use strict' | ||
|
||
const { workerData: { rcPort } } = require('node:worker_threads') | ||
const { getScript, probes, breakpoints } = require('./state') | ||
const session = require('./session') | ||
const { ackReceived, ackInstalled, ackError } = require('./status') | ||
const log = require('../../log') | ||
|
||
let sessionStarted = false | ||
|
||
// Example log line probe (simplified): | ||
// { | ||
// id: '100c9a5c-45ad-49dc-818b-c570d31e11d1', | ||
// version: 0, | ||
// type: 'LOG_PROBE', | ||
// where: { sourceFile: 'index.js', lines: ['25'] }, // only use first array element | ||
// template: 'Hello World 2', | ||
// segments: [...], | ||
// captureSnapshot: true, | ||
// capture: { maxReferenceDepth: 1 }, | ||
// sampling: { snapshotsPerSecond: 1 }, | ||
// evaluateAt: 'EXIT' // only used for method probes | ||
// } | ||
// | ||
// Example log method probe (simplified): | ||
// { | ||
// id: 'd692ee6d-5734-4df7-9d86-e3bc6449cc8c', | ||
// version: 0, | ||
// type: 'LOG_PROBE', | ||
// where: { typeName: 'index.js', methodName: 'handlerA' }, | ||
// template: 'Executed index.js.handlerA, it took {@duration}ms', | ||
// segments: [...], | ||
// captureSnapshot: false, | ||
// capture: { maxReferenceDepth: 3 }, | ||
// sampling: { snapshotsPerSecond: 5000 }, | ||
// 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})`) | ||
} | ||
if (!probe.where.sourceFile && !probe.where.lines) { | ||
throw new Error( | ||
// eslint-disable-next-line max-len | ||
`Unsupported probe insertion point! Only line-based probes are supported (id: ${probe.id}, version: ${probe.version})` | ||
) | ||
} | ||
|
||
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() | ||
|
||
const file = probe.where.sourceFile | ||
const line = Number(probe.where.lines[0]) // Tracer doesn't support multiple-line breakpoints | ||
|
||
// Optimize for sending data to /debugger/v1/input endpoint | ||
probe.location = { file, lines: [line] } | ||
delete probe.where | ||
|
||
const script = getScript(file) | ||
if (!script) throw new Error(`No loaded script found for ${file} (probe: ${probe.id}, version: ${probe.version})`) | ||
const [path, scriptId] = script | ||
|
||
log.debug(`Adding breakpoint at ${path}:${line} (probe: ${probe.id}, version: ${probe.version})`) | ||
|
||
const { breakpointId } = await session.post('Debugger.setBreakpoint', { | ||
location: { | ||
scriptId, | ||
lineNumber: line - 1 // Beware! lineNumber is zero-indexed | ||
} | ||
}) | ||
|
||
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 fail unexpected | ||
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,41 @@ | ||
'use strict' | ||
|
||
const { threadId } = require('node:worker_threads') | ||
const config = require('./config') | ||
const log = require('../../log') | ||
const request = require('../../exporters/common/request') | ||
|
||
module.exports = send | ||
|
||
const ddsource = 'dd_debugger' | ||
const service = config.service | ||
|
||
// TODO: Figure out correct logger values | ||
const logger = { | ||
name: __filename, // name of the class/type/file emitting the snapshot | ||
method: send.name, // name of the method/function emitting the snapshot | ||
version: 2, // version of the snapshot format (not currently used or enforced) | ||
thread_id: threadId, // current thread/process id emitting the snapshot | ||
thread_name: `${process.argv0};pid:${process.pid}` // name of the current thread emitting the snapshot | ||
} | ||
|
||
async function send (message, snapshot) { | ||
const opts = { | ||
method: 'POST', | ||
url: config.url, | ||
path: '/debugger/v1/input', | ||
headers: { 'Content-Type': 'application/json; charset=utf-8' } | ||
} | ||
|
||
const payload = { | ||
ddsource, | ||
service, | ||
message, | ||
logger, | ||
'debugger.snapshot': snapshot | ||
} | ||
|
||
request(JSON.stringify(payload), opts, (err) => { | ||
if (err) log.error(err) | ||
}) | ||
} |
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() |
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,46 @@ | ||
'use strict' | ||
|
||
const session = require('./session') | ||
|
||
const scripts = [] | ||
|
||
module.exports = { | ||
probes: new Map(), | ||
breakpoints: new Map(), | ||
|
||
/** | ||
* Find the matching script that can be inspected based on a partial path. | ||
* | ||
* Algorithm: Find the sortest url that ends in the requested path. | ||
* | ||
* Will identify the correct script as long as Node.js doesn't load a module from a `node_modules` folder outside the | ||
* project root. If so, there's a risk that this path is sorter than the expected path inside the project root. | ||
* Example of mismatch where path = `index.js`: | ||
* | ||
* Expected match: /www/code/my-projects/demo-project1/index.js | ||
* Actual sorter match: /www/node_modules/dd-trace/index.js | ||
* | ||
* To fix this, specify a more unique file path, e.g `demo-project1/index.js` instead of `index.js` | ||
* | ||
* @param {string} path | ||
* @returns {[string, string] | undefined} | ||
*/ | ||
getScript (path) { | ||
return scripts | ||
.filter(([url]) => url.endsWith(path)) | ||
.sort(([a], [b]) => a.length - b.length)[0] | ||
} | ||
} | ||
|
||
// Known params.url protocols: | ||
// - `node:` - Ignored, as we don't want to instrument Node.js internals | ||
// - `wasm:` - Ignored, as we don't support instrumenting WebAssembly | ||
// - `file:` - Regular on disk file | ||
// Unknown params.url values: | ||
// - `structured-stack` - Not sure what this is, but should just be ignored | ||
// - `` - Not sure what this is, but should just be ignored | ||
session.on('Debugger.scriptParsed', ({ params }) => { | ||
if (params.url.startsWith('file:')) { | ||
scripts.push([params.url, params.scriptId]) | ||
} | ||
}) |
Oops, something went wrong.