Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feature: sqs queue instrumentation #2013

Merged
merged 62 commits into from
Apr 7, 2021
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
25f0a73
feat: instrument amazon SQS
Mar 15, 2021
e2bfc9b
feat: received message span measures work done to process it, start o…
Mar 15, 2021
77c2e92
feat: configuration
Mar 17, 2021
869e326
feat: tests
Mar 18, 2021
3bf9daa
feat: integration tests
Mar 19, 2021
6852fdb
test: integration test for five main method calls
Mar 19, 2021
fb4dcd7
fix: lint
Mar 19, 2021
20c85b0
fix: sqs is always polling
Mar 19, 2021
e5fa3a4
docs: first draft of documentation
Mar 19, 2021
48bcb77
feat: logging for no transaction, adding semver check
Mar 19, 2021
96251a6
fix: AWS not mysql2
Mar 19, 2021
8eeb926
test: add tav configuration
Mar 19, 2021
29b26d7
docs: post-tech review for docs
Mar 22, 2021
29987f9
Merge branch 'master' into astorm/aws-sdk-sqs
Mar 23, 2021
e963940
fix: merge debris
Mar 23, 2021
63ca069
feat: metrics
Mar 23, 2021
8dca1c7
chore: doc blocks
Mar 23, 2021
0b89dc2
fix: pre-commit fixes
Mar 23, 2021
79bc833
fix: more doc blocks
Mar 23, 2021
a05f7b0
fix: lint, aws sqs request failures
Mar 23, 2021
527df68
fix: node 8 syntax issue
Mar 23, 2021
7d95562
Merge branch 'master' into astorm/aws-sdk-sqs
Mar 24, 2021
90ac7ce
Merge branch 'master' into astorm/aws-sdk-sqs
astorm Mar 26, 2021
2a7ec7c
Merge branch 'master' into astorm/aws-sdk-sqs
Mar 29, 2021
621c1bb
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
0417cf0
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
5677e47
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
df09585
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
b2ab2d3
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
423b65f
Update lib/instrumentation/modules/aws-sdk/sqs.js
astorm Mar 29, 2021
6abec2a
docs: expanded changelog
Mar 29, 2021
504e5e9
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
e67e733
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
de9ee64
chore: grammar
Mar 29, 2021
c35b6ea
chore: subtype
Mar 29, 2021
9851c66
chore: enabled guard
Mar 29, 2021
d38762d
Update docs/message-queues.asciidoc
astorm Mar 29, 2021
a88cc95
docs: remove confusing extra caveat
Mar 29, 2021
0bfefd7
docs: no such things as level 4 titles
Mar 29, 2021
2f8de38
docs: small rewrite of dt
Mar 29, 2021
cb56aa6
chore: flakey test
Mar 30, 2021
e14a6c9
feat: refactor to use bindFunction
Mar 30, 2021
49084cc
chore: test race condition
Mar 30, 2021
a3d3f2b
chore: any is a broad statement
Mar 30, 2021
7f2827f
Update lib/instrumentation/index.js
astorm Apr 1, 2021
058038e
Update lib/instrumentation/index.js
astorm Apr 1, 2021
337c3ea
Update lib/instrumentation/index.js
astorm Apr 1, 2021
1865b85
Update lib/instrumentation/index.js
astorm Apr 1, 2021
d7c4eee
Merge branch 'master' into astorm/aws-sdk-sqs
astorm Apr 1, 2021
02f6620
feat: shift span creation strat., adjust tests, ensure tests run in ci
Apr 1, 2021
7304c29
feat: docs, messaging context
Apr 1, 2021
3bc5a45
feat: adding support for .promise
Apr 1, 2021
9434579
feature: promise and tests for same
Apr 1, 2021
95a5d9b
feat: logger for tests, old age variable
Apr 1, 2021
0d111c8
fix: fixing test fixture to be message aware
Apr 1, 2021
c5603c1
chore: remove error logging
Apr 2, 2021
6c88f7f
chore: change to message instead of messaging
Apr 5, 2021
d99a68f
chore: update comment
Apr 5, 2021
2941bda
chore: unhardcode queue
Apr 5, 2021
e4eee59
chore: what's a Metics
Apr 6, 2021
4646ca0
dive bomb into alan's PR to see if I can temporarily fix tests
trentm Apr 6, 2021
b34cacf
Merge branch 'master' into astorm/aws-sdk-sqs
Apr 6, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions docs/supported-technologies.asciidoc
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ The Node.js agent will automatically instrument the following modules to give yo
[options="header"]
|=======================================================================
|Module |Version |Note
|https://www.npmjs.com/package/aws-sdk[aws-sdk] |>1 <3 |Will instrument SQS send/receive/delete messages
|https://www.npmjs.com/package/cassandra-driver[cassandra-driver] |>=3.0.0 |Will instrument all queries
|https://www.npmjs.com/package/elasticsearch[elasticsearch] |>=8.0.0 |Will instrument all queries
|https://www.npmjs.com/package/@elastic/elasticsearch[@elastic/elasticsearch] |>=7.0.0 <8.0.0 |Will instrument all queries
Expand Down
8 changes: 4 additions & 4 deletions lib/instrumentation/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -308,7 +308,7 @@ var wrapped = Symbol('elastic-apm-wrapped-function')
// Binds a callback function to the currently active span
//
// An instrumentation programmer can use this function to wrap a callback
// function of another function at the call-time or the original function.
// function of another function at the call-time of the original function.
// The pattern is
//
// 1. Instrumentation programmer uses shimmer.wrap to wrap a function that also
Expand All @@ -322,15 +322,15 @@ var wrapped = Symbol('elastic-apm-wrapped-function')
// argument when calling the original function.
//
// bindFunction function will "save" the currently active span via closure,
// and when the callback is invoked, the span and transaction active then
// and when the callback is invoked, the span and transaction active when
// the program called original function will be set as active. This ensures
// the callback function gets instrument on "the right" transaction and span.
//
// The instrumentation programmer is still respoisble for starting a span,
// The instrumentation programmer is still responsible for starting a span,
// and ending a span. Additionally, this function will set a span's sync
// property to `false` -- it's up to the instrumentation programmer to ensure
// that the callback they're binding is really async. If bindFunction is
// passed a callback that the wrapped function executes syncronously, it will
// passed a callback that the wrapped function executes synchronously, it will
// still mark the span's `async` property as `false`.
//
// @param {function} original
Expand Down
5 changes: 5 additions & 0 deletions lib/instrumentation/modules/aws-sdk.js
Original file line number Diff line number Diff line change
Expand Up @@ -31,5 +31,10 @@ module.exports = function (AWS, agent, { version, enabled }) {
}
})

shimmer.wrap(AWS.Request.prototype, 'promise', function (orig) {
return function _wrappedAWSRequestPromise () {
return instrumentOperation(orig, arguments, this, AWS, agent, { version, enabled })
}
})
return AWS
}
59 changes: 29 additions & 30 deletions lib/instrumentation/modules/aws-sdk/sqs.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
'use strict'

const { URL } = require('url')
const constants = require('../../../constants')
const OPERATIONS_TO_ACTIONS = {
deleteMessage: 'delete',
deleteMessageBatch: 'delete_batch',
Expand Down Expand Up @@ -70,6 +71,16 @@ function getMessagingDestinationContextFromRequest (request) {
return destination
}

// create message context suitable for setMessagingContext
function getMessagingContextFromRequest (request) {
const messaging = {
queue: {
name: getQueueNameFromRequest(request)
}
}
return messaging
}

// Record queue related metrics
//
// Creates metric collector objects on first run, and
Expand Down Expand Up @@ -127,16 +138,6 @@ function shouldIgnoreRequest (request, agent) {
return false
}

function wrapReceiveMessageCallback (original, request, agent) {
if (typeof original !== 'function' || original.name === 'elasticReceiveMessageCallback') return original

return elasticReceiveMessageCallback
function elasticReceiveMessageCallback (err, data) {
recordMetrics(getQueueNameFromRequest(request), data, agent)
return original.apply(this, [err, data])
}
}

// Main entrypoint for SQS instrumentation
//
// Must call (or one of its function calls must call) the
Expand All @@ -152,31 +153,29 @@ function instrumentationSqs (orig, origArguments, request, AWS, agent, { version
const name = getSpanNameFromRequest(request)
const span = agent.startSpan(name, type, subtype, action)
span.setDestinationContext(getMessagingDestinationContextFromRequest(request))
astorm marked this conversation as resolved.
Show resolved Hide resolved
span.setMessagingContext(getMessagingContextFromRequest(request))

const cb = origArguments[0] ? origArguments[0] : null
let wrappedCallback = agent._instrumentation.bindFunction(cb)
request.on('complete', function (response) {
if (response && response.error) {
const errOpts = {
skipOutcome: true
}
agent.captureError(response.error, errOpts)
span._setOutcomeFromErrorCapture(constants.OUTCOME_FAILURE)
}

// if we're receiving a message we need on more layer of
// wrapping in order to access the receiveMessage callback
// data and use that data to record metrics
if (OPERATIONS_TO_ACTIONS.receiveMessage === action) {
wrappedCallback = wrapReceiveMessageCallback(wrappedCallback, request, agent)
}
// we'll need to manually mark this span as async. The actual async hop
// is captured by the agent's async hooks instrumentation
span.sync = false
span.end()

// replace the argument
origArguments[0] = wrappedCallback
if (request.operation === 'receiveMessage' && response && response.data) {
recordMetrics(getQueueNameFromRequest(request), response.data, agent)
}
})

const origResult = orig.apply(request, origArguments)

// our complete listener MUST be registered after
// calling the original Request.send. Request.send
// invokes its callback function by registering its
// own complete callback. We need our complete callback
// to fire _after_ the one registered by Request.send
// see: https://github.com/aws/aws-sdk-js/blob/e5c12fad7ebc87b4de447d777628030be4792945/lib/request.js#L368
request.on('complete', function () {
span.end()
})
return origResult
}

Expand Down
10 changes: 8 additions & 2 deletions lib/instrumentation/span.js
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ function Span (transaction, name, ...args) {
this._db = null
this._http = null
this._destination = null
this._messaging = null
this._stackObj = null

this.transaction = transaction
Expand Down Expand Up @@ -96,6 +97,10 @@ Span.prototype.setDestinationContext = function (context) {
this._destination = Object.assign(this._destination || {}, context)
}

Span.prototype.setMessagingContext = function (context) {
astorm marked this conversation as resolved.
Show resolved Hide resolved
this._messaging = Object.assign(this._messaging || {}, context)
}

Span.prototype.setOutcome = function (outcome) {
if (!this._isValidOutcome(outcome)) {
this._agent.logger.trace(
Expand Down Expand Up @@ -214,12 +219,13 @@ Span.prototype._encode = function (cb) {
outcome: self.outcome
}

if (self._db || self._http || self._labels || self._destination) {
if (self._db || self._http || self._labels || self._destination || self._messaging) {
payload.context = {
db: self._db || undefined,
http: self._http || undefined,
tags: self._labels || undefined,
destination: self._destination || undefined
destination: self._destination || undefined,
message: self._messaging || undefined
}
}

Expand Down
Loading