Skip to content

Commit

Permalink
[pinpoint-apm#140] Support nested async trace
Browse files Browse the repository at this point in the history
* Remove no needs an argument TraceContext
  • Loading branch information
feelform committed Nov 22, 2023
1 parent 5db9558 commit fe1d911
Show file tree
Hide file tree
Showing 21 changed files with 166 additions and 7,934 deletions.
5 changes: 1 addition & 4 deletions lib/agent.js
Original file line number Diff line number Diff line change
Expand Up @@ -26,13 +26,10 @@ class Agent {

log.warn('[Pinpoint Agent] Configuration', this.config)

if (!this.config.enable || this.config.enable.toString() !== 'true') {
global.__PINPOINT_ENABLED__ = false
if (!this.config || !this.config.enable || this.config.enable.toString() !== 'true') {
log.warn('[Pinpoint Agent][' + this.config.agentId + '] Disabled')
return
}

global.__PINPOINT_ENABLED__ = true
log.warn('[Pinpoint Agent][' + this.config.agentId + '] Init Started')

const agentId = this.config.agentId
Expand Down
5 changes: 1 addition & 4 deletions lib/context/context-manager.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@
'use strict'

const async_hooks = require('async_hooks')

const traceObjectMap = new Map()

const init = (asyncId, type, triggerAsyncId, resource) => {

Check failure on line 12 in lib/context/context-manager.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'resource' is defined but never used. (no-unused-vars)

Check failure on line 12 in lib/context/context-manager.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'resource' is defined but never used. (no-unused-vars)

Check failure on line 12 in lib/context/context-manager.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'resource' is defined but never used. (no-unused-vars)
Expand All @@ -24,9 +23,7 @@ const destroy = (asyncId) => {
}

const start = () => {
if (global.__PINPOINT_ENABLED__) {
async_hooks.createHook({init, destroy}).enable()
}
async_hooks.createHook({init, destroy}).enable()
}

const getObject = () => {
Expand Down
18 changes: 17 additions & 1 deletion lib/instrumentation/context/local-storage.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,13 +9,18 @@
const semver = require('semver')
const AsyncContextLocalStorage = require('./async-context-local-storage')
const AsyncHooksLocalStorage = require('./async-hooks-local-storage')

const { getConfig } = require('../../config')
class LocalStorage {
constructor() {
this.storage = this.createLocalStorage()
}

createLocalStorage() {
const config = getConfig()
if (config && !config.enable) {
return new EmptyLocalStorage()
}

if (!semver.satisfies(process.version, '>=16.4.0')) {
return new AsyncHooksLocalStorage()
}
Expand All @@ -35,4 +40,15 @@ class LocalStorage {
}
}

class EmptyLocalStorage {
run() {
}

getStore() {
}

disable() {
}
}

module.exports = new LocalStorage()
11 changes: 1 addition & 10 deletions lib/instrumentation/instrument-method-context.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,22 +7,13 @@
'use strict'

class InstrumentMethodContext {
constructor(methodDescriptorBuilder, traceContext) {
constructor(methodDescriptorBuilder) {
this.methodDescriptorBuilder = methodDescriptorBuilder
this.traceContext = traceContext
}

getMethodDescriptorBuilder() {
return this.methodDescriptorBuilder
}

getTraceContext() {
return this.traceContext
}

currentTraceObject() {
return this.traceContext.currentTraceObject()
}
}

module.exports = InstrumentMethodContext
15 changes: 7 additions & 8 deletions lib/instrumentation/instrument-method.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,17 +13,17 @@ const { callSite } = require('./call-stack')
const InstrumentMethodContext = require('./instrument-method-context')
const config = require('../config')
const TraceBuilder = require('./context/trace-builder')
const localStorage = require('./context/local-storage')

// ref: SpanEventSimpleAroundInterceptorForPlugin.java
class InstrumentMethod {
constructor(target, method, traceContext) {
constructor(target, method) {
this.target = target
this.method = method
this.traceContext = traceContext
}

static make(target, method, traceContext) {
return new InstrumentMethod(target, method, traceContext)
static make(target, method) {
return new InstrumentMethod(target, method)
}

addScopedInterceptor(interceptor) {
Expand All @@ -35,7 +35,6 @@ class InstrumentMethod {
return
}

const traceContext = this.traceContext
const target = this.target
shimmer.wrap(this.target, this.method, function (original) {
return function () {
Expand All @@ -46,13 +45,13 @@ class InstrumentMethod {
} else {
builder = interceptor.methodDescriptorBuilder
}
const context = new InstrumentMethodContext(builder, traceContext)


const context = new InstrumentMethodContext(builder)
if (typeof interceptor.prepareBeforeTrace === 'function') {
interceptor.prepareBeforeTrace(target, arguments, this, context)
}

const trace = traceContext.currentTraceObject()
const trace = localStorage.getStore()
let recorder
if (trace) {
recorder = trace.traceBlockBegin()
Expand Down
4 changes: 2 additions & 2 deletions lib/instrumentation/module/express.js
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,8 @@ module.exports = function (agent, version, express) {
log.error('Invalid Version error')
return express
}
InstrumentMethod.make(express.Router, 'use', agent.traceContext).addScopedInterceptor(new ExpressUseInterceptor(agent.traceContext))
InstrumentMethod.make(express.Router, 'route', agent.traceContext).addScopedInterceptor(new ExpressRouteInterceptor(agent.traceContext))
InstrumentMethod.make(express.Router, 'use').addScopedInterceptor(new ExpressUseInterceptor())
InstrumentMethod.make(express.Router, 'route').addScopedInterceptor(new ExpressRouteInterceptor())

return express
}
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,7 @@ const MethodDescriptorBuilder = require('../../../context/method-descriptor-buil
const patchLayer = require('./patch-layer')

class ExpressRouteInterceptor {
constructor(traceContext) {
this.traceContext = traceContext
constructor() {
this.methodDescriptorBuilder = MethodDescriptorBuilder.makeRuntimeDetectionMethodDescriptorBuilder()
.setClassName('Router')
.setFileNameIndex(3)
Expand Down
5 changes: 3 additions & 2 deletions lib/instrumentation/module/express/patch-layer.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,13 +11,14 @@ const shimmer = require('@pinpoint-apm/shimmer')
const serviceType = require('./express-service-type')
const util = require('util')
const errorReportedSymbol = Symbol('PinpointErrorReported')
const localStorage = require('../../../instrumentation/context/local-storage')

function patchLayer(layer, context) {
shimmer.wrap(layer, 'handle', function (original) {
let handle
if (original.length !== 4) {
handle = function(req, res, next) {
const trace = context.currentTraceObject()
const trace = localStorage.getStore()
if (!trace) {
return original.apply(this, arguments)
}
Expand Down Expand Up @@ -45,7 +46,7 @@ function patchLayer(layer, context) {
}
} else {
handle = function (err, req, res, next) {

Check failure on line 48 in lib/instrumentation/module/express/patch-layer.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'res' is defined but never used. (no-unused-vars)

Check failure on line 48 in lib/instrumentation/module/express/patch-layer.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'next' is defined but never used. (no-unused-vars)

Check failure on line 48 in lib/instrumentation/module/express/patch-layer.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'res' is defined but never used. (no-unused-vars)

Check failure on line 48 in lib/instrumentation/module/express/patch-layer.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'next' is defined but never used. (no-unused-vars)

Check failure on line 48 in lib/instrumentation/module/express/patch-layer.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'res' is defined but never used. (no-unused-vars)

Check failure on line 48 in lib/instrumentation/module/express/patch-layer.js

View workflow job for this annotation

GitHub Actions / Checkstyle

eslint.rules.no-unused-vars

'next' is defined but never used. (no-unused-vars)
const trace = context.currentTraceObject()
const trace = localStorage.getStore()
let spanEventRecorder = null
if (shouldReport(err, trace)) {
spanEventRecorder = trace.traceBlockBegin()
Expand Down
2 changes: 1 addition & 1 deletion lib/instrumentation/module/koa-router.js
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ module.exports = function (agent, version, router) {
return router
}

InstrumentMethod.make(router.prototype, 'register', agent.traceContext).addScopedInterceptor(new KoaRegisterInterceptor())
InstrumentMethod.make(router.prototype, 'register').addScopedInterceptor(new KoaRegisterInterceptor())

return router
}
3 changes: 2 additions & 1 deletion lib/instrumentation/module/koa/koa-register-interceptor.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
const apiMetaService = require('../../../context/api-meta-service')
const MethodDescriptorBuilder = require('../../../context/method-descriptor-builder')
const serviceType = require('./koa-service-type')
const localStorage = require('../../../instrumentation/context/local-storage')

class KoaRegisterInterceptor {
constructor() {
Expand Down Expand Up @@ -45,7 +46,7 @@ class KoaRegisterInterceptor {
builder = builder.makeCloneOf(method.toLowerCase())
}
const methodDescriptor = apiMetaService.cacheApiWithBuilder(builder)
const trace = context.currentTraceObject()
const trace = localStorage.getStore()
let spanEventRecorder = null
let result
try {
Expand Down
6 changes: 3 additions & 3 deletions lib/instrumentation/module/mysql.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,13 @@ module.exports = function (agent, version, mysql) {
}

log.debug('shimming mysql.createPool')
InstrumentMethod.make(mysql, 'createPool', agent.traceContext).addScopedInterceptor(new MySQLCreatePoolInterceptor(agent.traceContext))
InstrumentMethod.make(mysql, 'createPool').addScopedInterceptor(new MySQLCreatePoolInterceptor())

log.debug('shimming mysql.createPoolCluster')
InstrumentMethod.make(mysql, 'createPoolCluster', agent.traceContext).addScopedInterceptor(new MySQLCreatePoolClusterInterceptor(agent.traceContext))
InstrumentMethod.make(mysql, 'createPoolCluster').addScopedInterceptor(new MySQLCreatePoolClusterInterceptor())

log.debug('shimming mysql.createConnection')
InstrumentMethod.make(mysql, 'createConnection', agent.traceContext).addScopedInterceptor(new MySQLCreateConnectionInterceptor(agent.traceContext))
InstrumentMethod.make(mysql, 'createConnection').addScopedInterceptor(new MySQLCreateConnectionInterceptor())

return mysql
}
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,7 @@ const InstrumentMethod = require('../../instrument-method')
const MySQLGetConnectionInterceptor = require('./mysql-get-connection-interceptor')

class MySQLClusterOfInterceptor {
constructor(traceContext) {
this.traceContext = traceContext
constructor() {
this.methodDescriptorBuilder = new MethodDescriptorBuilder('of')
}

Expand All @@ -24,7 +23,7 @@ class MySQLClusterOfInterceptor {
if (!poolNamespace) {
return
}
InstrumentMethod.make(poolNamespace, 'getConnection', this.traceContext).addScopedInterceptor(new MySQLGetConnectionInterceptor(this.traceContext))
InstrumentMethod.make(poolNamespace, 'getConnection').addScopedInterceptor(new MySQLGetConnectionInterceptor())
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,9 @@ const databaseInfoSymbol = require('./mysql-database-information-symbol')
const MySQLDatabaseInformationContext = require('./mysql-database-information-context')

class MySQLCreateConnectionInterceptor {
constructor(traceContext) {
this.traceContext = traceContext
constructor() {
this.methodDescriptorBuilder = new MethodDescriptorBuilder('createConnection')
this.methodDescriptorBuilder.setParameterVariableNames('connectionUri')
.setParameterVariableNames('connectionUri')
}

doInBeforeTrace(recorder) {
Expand All @@ -28,8 +27,8 @@ class MySQLCreateConnectionInterceptor {
doInAfterTrace(recorder, target, args, returned) {
if (returned) {
returned[databaseInfoSymbol] = MySQLDatabaseInformationContext.recordAndReturnDatabaseInfo(recorder, target, args)
InstrumentMethod.make(returned, 'connect', this.traceContext).addScopedInterceptor(new MySQLConnectionInterceptor())
InstrumentMethod.make(returned, 'query', this.traceContext).addScopedInterceptor(new MySQLStatementExecuteQueryInterceptor('Connection'))
InstrumentMethod.make(returned, 'connect').addScopedInterceptor(new MySQLConnectionInterceptor())
InstrumentMethod.make(returned, 'query').addScopedInterceptor(new MySQLStatementExecuteQueryInterceptor('Connection'))
}
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,7 @@ const InstrumentMethod = require('../../instrument-method')
const MySQLClusterOfInterceptor = require('./mysql-cluster-of-interceptor')

class MySQLCreatePoolClusterInterceptor {
constructor(traceContext) {
this.traceContext = traceContext
constructor() {
this.methodDescriptorBuilder = new MethodDescriptorBuilder('createPoolCluster')
}

Expand All @@ -24,7 +23,7 @@ class MySQLCreatePoolClusterInterceptor {
if (!poolCluster) {
return
}
InstrumentMethod.make(poolCluster, 'of', this.traceContext).addScopedInterceptor(new MySQLClusterOfInterceptor(this.traceContext))
InstrumentMethod.make(poolCluster, 'of').addScopedInterceptor(new MySQLClusterOfInterceptor())
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,7 @@ const MySQLGetConnectionInterceptor = require('./mysql-get-connection-intercepto
const MySQLDatabaseInformationContext = require('./mysql-database-information-context')

class MySQLCreatePoolInterceptor {
constructor(traceContext) {
this.traceContext = traceContext
constructor() {
this.methodDescriptorBuilder = new MethodDescriptorBuilder('createPool')
}

Expand All @@ -25,7 +24,7 @@ class MySQLCreatePoolInterceptor {

prepareAfterTrace(target, args, pool) {
if (pool) {
InstrumentMethod.make(pool, 'getConnection', this.traceContext).addScopedInterceptor(new MySQLGetConnectionInterceptor(this.traceContext))
InstrumentMethod.make(pool, 'getConnection').addScopedInterceptor(new MySQLGetConnectionInterceptor())
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,7 @@ const InstrumentMethod = require('../../instrument-method')
const MySQLStatementExecuteQueryInterceptor = require('./mysql-statement-execute-query-interceptor')

class MySQLGetConnectionInterceptor {
constructor(traceContext) {
this.traceContext = traceContext
constructor() {
this.methodDescriptorBuilder = new MethodDescriptorBuilder('getConnection')
.setClassName('Pool')
this.serviceType = serviceType
Expand All @@ -31,7 +30,7 @@ class MySQLGetConnectionInterceptor {
prepareBeforeAsyncTrace(target, args) {
if (args.length > 1 && args[1]) {
const connection = args[1]
InstrumentMethod.make(connection, 'query', this.traceContext).addScopedInterceptor(new MySQLStatementExecuteQueryInterceptor('PoolConnection'))
InstrumentMethod.make(connection, 'query').addScopedInterceptor(new MySQLStatementExecuteQueryInterceptor('PoolConnection'))
}
}

Expand Down
6 changes: 3 additions & 3 deletions lib/instrumentation/module/mysql2.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,13 @@ module.exports = function (agent, version, mysql) {
}

log.debug('shimming mysql.createPool')
InstrumentMethod.make(mysql, 'createPool', agent.traceContext).addScopedInterceptor(new MySQLCreatePoolInterceptor(agent.traceContext))
InstrumentMethod.make(mysql, 'createPool').addScopedInterceptor(new MySQLCreatePoolInterceptor())

log.debug('shimming mysql.createPoolCluster')
InstrumentMethod.make(mysql, 'createPoolCluster', agent.traceContext).addScopedInterceptor(new MySQLCreatePoolClusterInterceptor(agent.traceContext))
InstrumentMethod.make(mysql, 'createPoolCluster').addScopedInterceptor(new MySQLCreatePoolClusterInterceptor())

log.debug('shimming mysql.createConnection')
InstrumentMethod.make(mysql, 'createConnection', agent.traceContext).addScopedInterceptor(new MySQLCreateConnectionInterceptor(agent.traceContext))
InstrumentMethod.make(mysql, 'createConnection').addScopedInterceptor(new MySQLCreateConnectionInterceptor())

return mysql
}
Loading

0 comments on commit fe1d911

Please sign in to comment.