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

feat(rum-core): capture XHR/Fetch spans using resource timing #825

Merged
merged 10 commits into from
Jul 1, 2020
4 changes: 2 additions & 2 deletions packages/rum-core/src/common/constants.js
Original file line number Diff line number Diff line change
Expand Up @@ -40,12 +40,12 @@ const REMOVE_EVENT_LISTENER_STR = 'removeEventListener'
* Resource Timing initiator types that will be captured as spans
*/
const RESOURCE_INITIATOR_TYPES = [
'xmlhttprequest',
'fetch',
'link',
'css',
'script',
'img',
'xmlhttprequest',
'fetch',
'beacon',
'iframe'
]
Expand Down
7 changes: 4 additions & 3 deletions packages/rum-core/src/common/patching/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,12 +28,13 @@ import { patchFetch } from './fetch-patch'
import { patchHistory } from './history-patch'
import { patchEventTarget } from './event-target-patch'
import EventHandler from '../event-handler'

import { now } from '../utils'
import { HISTORY, FETCH, XMLHTTPREQUEST, EVENT_TARGET } from '../constants'
import { state } from '../../state'

const patchEventHandler = new EventHandler()

let alreadyPatched = false

function patchAll() {
if (!alreadyPatched) {
alreadyPatched = true
Expand All @@ -46,10 +47,10 @@ function patchAll() {
patchHistory(function(event, task) {
patchEventHandler.send(HISTORY, [event, task])
})

patchEventTarget(function(event, task) {
patchEventHandler.send(EVENT_TARGET, [event, task])
})
state.patchedTime = now()
vigneshshanmugam marked this conversation as resolved.
Show resolved Hide resolved
}
return patchEventHandler
}
Expand Down
102 changes: 43 additions & 59 deletions packages/rum-core/src/performance-monitoring/capture-navigation.js
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ import {
PERF,
isPerfTimelineSupported
} from '../common/utils'
import { state } from '../state'

/**
* Navigation Timing Spans
Expand Down Expand Up @@ -119,60 +120,58 @@ function createResourceTimingSpan(resourceTimingEntry) {
return span
}

function createResourceTimingSpans(entries, filterUrls, trStart, trEnd) {
/**
* Checks if the span is already captured via XHR/Fetch patch by
* comparing the given resource startTime(fetchStart) aganist the
* patch code execution time.
*/
function isCapturedByPatching(resourceStartTime, requestPatchTime) {
return requestPatchTime != null && resourceStartTime > requestPatchTime
}

/**
* Check if the given url matches APM Server's Intake
* API endpoint and ignore it from Spans
*/
function isIntakeAPIEndpoint(url) {
return /intake\/v\d+\/rum\/events/.test(url)
vigneshshanmugam marked this conversation as resolved.
Show resolved Hide resolved
}

function createResourceTimingSpans(entries, requestPatchTime, trStart, trEnd) {
const spans = []
for (let i = 0; i < entries.length; i++) {
let { initiatorType, name, startTime, responseEnd } = entries[i]
const { initiatorType, name, startTime, responseEnd } = entries[i]
/**
* Skipping the timing information of API calls because of auto patching XHR and Fetch
* Skip span creation if initiatorType is other than known types specified as part of RESOURCE_INITIATOR_TYPES
* The reason being, there are other types like embed, video, audio, navigation etc
*
* Check the below webplatform test to know more
* https://github.com/web-platform-tests/wpt/blob/b0020d5df18998609b38786878f7a0b92cc680aa/resource-timing/resource_initiator_types.html#L93
*/
if (
initiatorType === 'xmlhttprequest' ||
initiatorType === 'fetch' ||
!name
RESOURCE_INITIATOR_TYPES.indexOf(initiatorType) === -1 ||
name == null
) {
continue
}

/**
* Create spans for all known resource initiator types
* Create Spans for API calls (XHR, Fetch) only if its not captured by the patch
*
* This would happen if our agent is downlaoded asyncrhonously and page does
* API requests before the agent patches the required modules.
*/
if (RESOURCE_INITIATOR_TYPES.indexOf(initiatorType) !== -1) {
if (!shouldCreateSpan(startTime, responseEnd, trStart, trEnd)) {
continue
}
spans.push(createResourceTimingSpan(entries[i]))
} else {
/**
* Since IE does not support initiatorType in Resource timing entry,
* We have to manually filter the API calls from creating duplicate Spans
*
* Skip span creation if initiatorType is other than known types specified as part of RESOURCE_INITIATOR_TYPES
* The reason being, there are other types like embed, video, audio, navigation etc
*
* Check the below webplatform test to know more
* https://github.com/web-platform-tests/wpt/blob/b0020d5df18998609b38786878f7a0b92cc680aa/resource-timing/resource_initiator_types.html#L93
*/
if (initiatorType != null) {
vigneshshanmugam marked this conversation as resolved.
Show resolved Hide resolved
continue
}
if (
(initiatorType === RESOURCE_INITIATOR_TYPES[0] ||
initiatorType === RESOURCE_INITIATOR_TYPES[1]) &&
vigneshshanmugam marked this conversation as resolved.
Show resolved Hide resolved
(isIntakeAPIEndpoint(name) ||
isCapturedByPatching(startTime, requestPatchTime))
) {
continue
}

let foundAjaxReq = false
for (let j = 0; j < filterUrls.length; j++) {
const idx = name.lastIndexOf(filterUrls[j])
if (idx > -1 && idx === name.length - filterUrls[j].length) {
foundAjaxReq = true
break
}
}
/**
* Create span if its not an ajax request
*/
if (
!foundAjaxReq &&
shouldCreateSpan(startTime, responseEnd, trStart, trEnd)
) {
spans.push(createResourceTimingSpan(entries[i]))
}
if (shouldCreateSpan(startTime, responseEnd, trStart, trEnd)) {
spans.push(createResourceTimingSpan(entries[i]))
}
}
return spans
Expand Down Expand Up @@ -200,18 +199,6 @@ function createUserTimingSpans(entries, trStart, trEnd) {
return userTimingSpans
}

function getApiSpanNames({ spans }) {
const apiCalls = []

for (let i = 0; i < spans.length; i++) {
const span = spans[i]
if (span.type === 'external' && span.subtype === 'http') {
apiCalls.push(span.name.split(' ')[1])
}
}
return apiCalls
}

/**
* Navigation timing marks are reported only for page-load transactions
*
Expand Down Expand Up @@ -312,7 +299,6 @@ function captureNavigation(transaction) {
* for few extra spans than soft navigations which
* happens on single page applications
*/

if (transaction.type === PAGE_LOAD) {
/**
* Adjust custom marks properly to fit in the transaction timeframe
Expand Down Expand Up @@ -356,11 +342,9 @@ function captureNavigation(transaction) {
* Capture resource timing information as spans
*/
const resourceEntries = PERF.getEntriesByType(RESOURCE)
const apiCalls = getApiSpanNames(transaction)

createResourceTimingSpans(
resourceEntries,
apiCalls,
state.patchedTime,
trStart,
trEnd
).forEach(span => transaction.spans.push(span))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,8 +27,8 @@ import {
checkSameOrigin,
isDtHeaderValid,
parseDtHeaderValue,
stripQueryStringFromUrl,
getDtHeaderValue
getDtHeaderValue,
stripQueryStringFromUrl
} from '../common/utils'
import Url from '../common/url'
import { patchEventHandler } from '../common/patching'
Expand Down
10 changes: 7 additions & 3 deletions packages/rum-core/src/state.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,11 @@
*/

const __DEV__ = process.env.NODE_ENV !== 'production'
const state = {
// Time when agent code is executed and patching of modules happens
patchedTime: null,
// Time when the document is last backgrounded
lastHiddenStart: Number.MIN_SAFE_INTEGER
hmdhk marked this conversation as resolved.
Show resolved Hide resolved
}

export { __DEV__ }

export const state = {}
export { __DEV__, state }
Loading