Skip to content

Commit

Permalink
Add newrelic.startSegment
Browse files Browse the repository at this point in the history
  • Loading branch information
NatalieWolfe authored and Peter Svetlichny committed Mar 26, 2018
1 parent dfbd589 commit 5c8dc44
Show file tree
Hide file tree
Showing 7 changed files with 173 additions and 6 deletions.
78 changes: 76 additions & 2 deletions api.js
Original file line number Diff line number Diff line change
Expand Up @@ -629,13 +629,23 @@ API.prototype.getBrowserTimingHeader = function getBrowserTimingHeader() {
return out
}

API.prototype.createTracer = util.deprecate(
createTracer, [
'API#createTracer is being deprecated!',
'Please use API#startSegment for segment creation.'
].join(' ')
)

/**
* This creates a new tracer with the passed in name. It then wraps the
* callback and binds it to the current transaction and segment so any further
* custom instrumentation as well as auto instrumentation will also be able to
* find the current transaction and segment.
*
* @memberof API#
* @deprecated use {@link API#startSegment} instead
*/
API.prototype.createTracer = function createTracer(name, callback) {
function createTracer(name, callback) {
var metric = this.agent.metrics.getOrCreateMetric(
NAMES.SUPPORTABILITY.API + '/createTracer'
)
Expand Down Expand Up @@ -687,6 +697,70 @@ API.prototype.createTracer = function createTracer(name, callback) {
return arity.fixArity(callback, tracer.bindFunction(callback, segment, true))
}

/**
* Wraps the given handler in a segment which may optionally be turned into a
* metric.
*
* @example
* newrelic.startSegment('mySegment', false, function handler() {
* // The returned promise here will signify the end of the segment.
* return myAsyncTask().then(myNextTask)
* })
*
* @param {string} name
* The name to give the new segment. This will also be the name of the metric.
*
* @param {bool} record
* Indicates if the segment should be recorded as a metric. Metrics will show
* up on the transaction breakdown table and server breakdown graph. Segments
* just show up in transaction traces.
*
* @param {function(cb) -> ?Promise} handler
* The function to track as a segment.
*
* @param {function} [callback]
* An optional callback for the handler. This will indicate the end of the
* timing if provided.
*
* @return {*} Returns the result of calling `handler`.
*/
API.prototype.startSegment = function startSegment(name, record, handler, callback) {
this.agent.metrics.getOrCreateMetric(
NAMES.SUPPORTABILITY.API + '/startSegment'
).incrementCallCount()

// Check that we have usable arguments.
if (!name || typeof handler !== 'function') {
logger.warn('Name and handler function are both required for startSegment')
if (typeof handler === 'function') {
return handler(callback)
}
return
}
if (callback && typeof callback !== 'function') {
logger.warn('If using callback, it must be a function')
return handler(callback)
}

// Are we inside a transaction?
if (!this.shim.getActiveSegment()) {
logger.debug('startSegment(%j) called outside of a transaction, not recording.', name)
return handler(callback)
}

// Create the segment and call the handler.
var wrappedHandler = this.shim.record(handler, function handlerNamer(shim) {
return {
name: name,
recorder: record ? customRecorder : null,
callback: callback ? shim.FIRST : null,
promise: !!callback
}
})

return wrappedHandler(callback)
}

API.prototype.createWebTransaction = util.deprecate(
createWebTransaction, [
'API#createWebTransaction is being deprecated!',
Expand All @@ -713,7 +787,7 @@ API.prototype.createWebTransaction = util.deprecate(
name and not iclude any variable parameters.
* @param {Function} handle Function that represents the transaction work.
*
* @memberOf API#
* @memberof API#
*
* @deprecated since version 2.0
*/
Expand Down
2 changes: 1 addition & 1 deletion examples/api/background-transactions/example1-basic.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ var newrelic = require('newrelic')

var transactionName = 'myCustomTransaction'

// startBackgroundTransaction() takes a name, group, and a handler function to
// `startBackgroundTransaction()` takes a name, group, and a handler function to
// execute. The group is optional. The last parameter is the function performing
// the work inside the transaction. Once the transaction starts, there are
// three ways to end it:
Expand Down
33 changes: 33 additions & 0 deletions examples/api/segments/example1-callbacks.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
'use strict'

var newrelic = require('newrelic')

// Segments can only be created inside of transactions. They could be automatically
// generated HTTP transactions or custom transactions.
newrelic.startBackgroundTransaction('bg-tx', function transHandler() {
var tx = newrelic.getTransaction()

// `startSegment()` takes a segment name, a boolean if a metric should be
// created for this segment, the handler function, and an optional callback.
// The handler is the function that will be wrapped with the new segment. When
// a callback is provided, the segment timing will end when the callback is
// called.

newrelic.startSegment('myCustomSegment', false, someTask, function cb(err, output) {
// Handle the error and output as appropriate.
console.log(output)
tx.end()
})
})

function someTask(cb) {
myAsyncTask(function firstCb(err, result) {
if (err) {
return cb(err)
}

myNextTask(result, function secondCb(err, output) {
cb(err, output)
})
})
}
24 changes: 24 additions & 0 deletions examples/api/segments/example2-promises.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
'use strict'

var newrelic = require('newrelic')

// Segments can only be created inside of transactions. They could be automatically
// generated HTTP transactions or custom transactions.
newrelic.startBackgroundTransaction('bg-tx', function transHandler() {
// `startSegment()` takes a segment name, a boolean if a metric should be
// created for this segment, the handler function, and an optional callback.
// The handler is the function that will be wrapped with the new segment. If
// a promise is returned from the handler, the segment's ending will be tied
// to that promise resolving or rejecting.

return newrelic.startSegment('myCustomSegment', false, someTask)
.then(function thenAfter(output) {
console.log(output)
})
})

function someTask() {
return myAsyncTask().then(function thenNext(result) {
return myNextTask(result)
})
}
22 changes: 22 additions & 0 deletions examples/api/segments/example3-async.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
'use strict'

var newrelic = require('newrelic')

// Segments can only be created inside of transactions. They could be automatically
// generated HTTP transactions or custom transactions.
newrelic.startBackgroundTransaction('bg-tx', async function transHandler() {
// `startSegment()` takes a segment name, a boolean if a metric should be
// created for this segment, the handler function, and an optional callback.
// The handler is the function that will be wrapped with the new segment.
// Since `async` functions just return a promise, they are covered just the
// same as the promise example.

var output = await newrelic.startSegment('myCustomSegment', false, someTask)
console.log(output)
})

async function someTask() {
var result = await myAsyncTask()
var output = await myNextTask(result)
return output
}
16 changes: 15 additions & 1 deletion stub_api.js
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,12 @@ for (var i = 0; i < length; i++) {
Stub.prototype[functionName] = stubFunction(functionName)
}

Stub.prototype.createTracer = createTracer
Stub.prototype.createTracer = util.deprecate(
createTracer, [
'API#createTracer is being deprecated!',
'Please use API#startSegment for segment creation.'
].join(' ')
)
Stub.prototype.createWebTransaction = util.deprecate(
createWebTransaction, [
'API#createWebTransaction is being deprecated!',
Expand All @@ -47,6 +52,7 @@ Stub.prototype.createBackgroundTransaction = util.deprecate(
'ending transactions.'
].join(' ')
)
Stub.prototype.startSegment = startSegment
Stub.prototype.startWebTransaction = startWebTransaction
Stub.prototype.startBackgroundTransaction = startBackgroundTransaction
Stub.prototype.getTransaction = getTransaction
Expand Down Expand Up @@ -81,6 +87,14 @@ function createBackgroundTransaction(name, group, callback) {
return (callback === undefined) ? group : callback
}

function startSegment(name, record, handler, callback) {
logger.debug('Not calling `startSegment` becuase New Relic is disabled.')
if (typeof handler === 'function') {
return handler(callback)
}
return null
}

function startWebTransaction(url, callback) {
logger.debug('Not calling startWebTransaction because New Relic is disabled.')
if (typeof callback === 'function') {
Expand Down
4 changes: 2 additions & 2 deletions test/unit/api/stub.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,8 @@ describe("the stubbed New Relic agent API", function() {
api = new API()
})

it("should export 27 API calls", function() {
expect(Object.keys(api.constructor.prototype).length).to.equal(27)
it("should export 28 API calls", function() {
expect(Object.keys(api.constructor.prototype).length).to.equal(28)
})

it("exports a transaction naming function", function() {
Expand Down

0 comments on commit 5c8dc44

Please sign in to comment.