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

✨ [RUMF-1175] collect reports and csp violation #1332

Merged
merged 26 commits into from
Mar 4, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
2f3b989
✨ Create report observable
amortemousque Feb 21, 2022
b391c17
Add reports in Log browser SDK
amortemousque Feb 21, 2022
715c46b
Add report in RUM browser SDK
amortemousque Feb 21, 2022
fcd4950
Add new forwardReports config option
amortemousque Feb 25, 2022
f74020d
Move report api stubs in its own file
amortemousque Mar 1, 2022
37180c2
Rum track report under FF
amortemousque Mar 1, 2022
43ef98b
Fix configuration validation
amortemousque Mar 1, 2022
875cdc3
Add tests
amortemousque Mar 1, 2022
e28db68
👌 Update browser support
amortemousque Mar 1, 2022
2cdc941
👌 use toStackTraceString
amortemousque Mar 2, 2022
998a897
Fix report init option allowedValues check
amortemousque Mar 2, 2022
e248579
Send report id instead of type + add more info for non error report
amortemousque Mar 2, 2022
1049d11
👌 Simplify report types
amortemousque Mar 2, 2022
5555282
👌 Add report test when init option is not specified
amortemousque Mar 2, 2022
c74dfd2
👌 Rename CustomReport to RawReport
amortemousque Mar 2, 2022
5f2f273
Merge branch 'main' into aymeric/collect-reports-and-csp-violation
amortemousque Mar 2, 2022
7c9a4f9
Update browser support
amortemousque Mar 2, 2022
3b5571d
👌 Report should be UNHANDLED
amortemousque Mar 2, 2022
9b236d8
👌 Add test on getFileFromStackTraceString
amortemousque Mar 2, 2022
a6f592f
Report id becomes subtype
amortemousque Mar 2, 2022
885f8e2
👌 Add policy to the stack message
amortemousque Mar 3, 2022
ac86464
Merge branch 'main' into aymeric/collect-reports-and-csp-violation
amortemousque Mar 3, 2022
fe11c96
Fix eslint issue + update format
amortemousque Mar 3, 2022
18d6978
Merge branch 'main' into aymeric/collect-reports-and-csp-violation
amortemousque Mar 3, 2022
2079e55
Fix ie compatibility
amortemousque Mar 4, 2022
7124d93
Merge branch 'main' into aymeric/collect-reports-and-csp-violation
amortemousque Mar 4, 2022
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
43 changes: 43 additions & 0 deletions packages/core/src/domain/report/browser.types.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
export interface BrowserWindow {
ReportingObserver?: ReportingObserver
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
}

export interface ReportingObserver {
disconnect(): void
observe(): void
takeRecords(): Report[]
}

export type ReportType = 'intervention' | 'deprecation'

export interface Report {
type: ReportType
url: string
body: DeprecationReportBody | InterventionReportBody
}

export interface ReportingObserverCallback {
(reports: Report[], observer: ReportingObserver): void
}

export interface ReportingObserverOption {
types: ReportType[]
buffered: boolean
}

interface DeprecationReportBody {
id: string
message: string
lineNumber: number
columnNumber: number
sourceFile: string
anticipatedRemoval?: Date
}

interface InterventionReportBody {
id: string
message: string
lineNumber: number
columnNumber: number
sourceFile: string
}
63 changes: 63 additions & 0 deletions packages/core/src/domain/report/reportObservable.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
import { isChromium } from '../../tools/browserDetection'
import type { Subscription } from '../../tools/observable'
import { stubReportingObserver, stubCspEventListener } from '../../../test/specHelper'
import { initReportObservable, CustomReportType } from './reportObservable'

describe(`report observable`, () => {
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
let reportingObserverStub: { reset(): void; raiseReport(type: string): void }
let cspEventListenerStub: { dispatchEvent(): void }
let consoleSubscription: Subscription
let notifyReport: jasmine.Spy

beforeEach(() => {
if (!isChromium()) {
pending('no ReportingObserver support')
}
amortemousque marked this conversation as resolved.
Show resolved Hide resolved

reportingObserverStub = stubReportingObserver()
cspEventListenerStub = stubCspEventListener()
notifyReport = jasmine.createSpy('notifyReport')
})

afterEach(() => {
reportingObserverStub.reset()
consoleSubscription.unsubscribe()
})
;[CustomReportType.deprecation, CustomReportType.deprecation].forEach((type) => {
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
bcaudan marked this conversation as resolved.
Show resolved Hide resolved
it(`should notify ${type} reports`, () => {
consoleSubscription = initReportObservable([type]).subscribe(notifyReport)
reportingObserverStub.raiseReport(type)

const [report] = notifyReport.calls.mostRecent().args

expect(report).toEqual(
jasmine.objectContaining({
message: `${type}: foo bar`,
type,
})
)
})
})

it(`should compute stack for ${CustomReportType.intervention}`, () => {
consoleSubscription = initReportObservable([CustomReportType.intervention]).subscribe(notifyReport)
reportingObserverStub.raiseReport(CustomReportType.intervention)

const [report] = notifyReport.calls.mostRecent().args

expect(report.stack).toEqual(`NavigatorVibrate: foo bar
at <anonymous> @ http://foo.bar/index.js:20:10`)
})

it(`should notify ${CustomReportType.csp_violation}`, () => {
consoleSubscription = initReportObservable([CustomReportType.csp_violation]).subscribe(notifyReport)
cspEventListenerStub.dispatchEvent()

expect(notifyReport).toHaveBeenCalledOnceWith({
message: `csp_violation: 'blob' blocked by 'worker-src' directive`,
type: 'csp_violation',
stack: `csp_violation: 'blob' blocked by 'worker-src' directive
at <anonymous> @ http://foo.bar/index.js:17:8`,
})
})
})
119 changes: 119 additions & 0 deletions packages/core/src/domain/report/reportObservable.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
import { mergeObservables, Observable } from '../../tools/observable'
import { DOM_EVENT, includes, addEventListener } from '../../tools/utils'
import { monitor } from '../internalMonitoring'
import type {
Report,
BrowserWindow,
ReportType,
ReportingObserverCallback,
ReportingObserverOption,
ReportingObserver as ReportingObserverInterface,
} from './browser.types'

declare const ReportingObserver: {
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
new (callback: ReportingObserverCallback, option: ReportingObserverOption): ReportingObserverInterface
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
}

export const CustomReportType = {
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
intervention: 'intervention',
deprecation: 'deprecation',
csp_violation: 'csp_violation',
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
} as const

export type CustomReportType = typeof CustomReportType[keyof typeof CustomReportType]

export interface CustomReport {
type: CustomReportType
message: string
stack?: string
}

export function initReportObservable(apis: CustomReportType[]) {
const observables: Array<Observable<CustomReport>> = []

if (includes(apis, CustomReportType.csp_violation)) {
observables.push(createCspViolationReportObservable())
}

const reportTypes = apis.filter((api: CustomReportType): api is ReportType => api !== CustomReportType.csp_violation)
if (reportTypes.length) {
observables.push(createReportObservable(reportTypes))
}

return mergeObservables<CustomReport>(...observables)
}

function createReportObservable(reportTypes: ReportType[]) {
const observable = new Observable<CustomReport>(() => {
if (!(window as BrowserWindow).ReportingObserver) {
return
}
const handleReports = monitor((reports: Report[]) =>
reports.forEach((report) => {
observable.notify(buildCustomReportFromReport(report))
})
)

const observer = new ReportingObserver(handleReports, {
types: reportTypes,
buffered: true,
})

observer.observe()
return () => {
observer.disconnect()
}
})

return observable
}

function createCspViolationReportObservable() {
const observable = new Observable<CustomReport>(() => {
const handleCspViolation = monitor((event: SecurityPolicyViolationEvent) => {
observable.notify(buildCustomReportFromCspViolation(event))
})

const { stop } = addEventListener(document, DOM_EVENT.SECURITY_POLICY_VIOLATION, handleCspViolation)

return stop
})
return observable
}

function buildCustomReportFromReport({
type,
body: { id, message, sourceFile, lineNumber, columnNumber },
}: Report): CustomReport {
const report: CustomReport = { type, message: `${type}: ${message}` }
if (type === CustomReportType.intervention) {
report.stack = buildStack(id, message, sourceFile, lineNumber, columnNumber)
}
return report
}

function buildCustomReportFromCspViolation(event: SecurityPolicyViolationEvent): CustomReport {
const type = 'csp_violation'
const message = `'${event.blockedURI}' blocked by '${event.effectiveDirective}' directive`
Copy link
Contributor Author

@amortemousque amortemousque Mar 2, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

After some thought, I think it is valuable to add the originalPolicy to the crafted message.
Could be done like so:
${event.blockedURI}' blocked by '${event.effectiveDirective}' directive of the policy "${event.originalPolicy}"

This way we will have message like:

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some directives could list a large number of domains 🤔
I'd say that it could be added in the stack to keep a minimal message.
We should maybe also truncate the policy to avoid sending too much data.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added it to the stack. Let me know what you think

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM!


return {
type,
message: `${type}: ${message}`,
stack: buildStack(type, message, event.sourceFile, event.lineNumber, event.columnNumber),
}
}

function buildStack(
type: string,
message: string,
sourceFile: string | null,
lineNumber: number | null,
columnNumber: number | null
): string | undefined {
if (!sourceFile) {
return undefined
}

return `${type}: ${message}
at <anonymous> @ ${[sourceFile, lineNumber, columnNumber].filter((e) => e).join(':')}`
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
}
1 change: 1 addition & 0 deletions packages/core/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ export {
export { trackRuntimeError } from './domain/error/trackRuntimeError'
export { computeStackTrace, StackTrace } from './domain/tracekit'
export { defineGlobal, makePublicApi } from './boot/init'
export { initReportObservable, CustomReport, CustomReportType } from './domain/report/reportObservable'
export {
startInternalMonitoring,
InternalMonitoring,
Expand Down
1 change: 1 addition & 0 deletions packages/core/src/tools/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ export const enum DOM_EVENT {
INPUT = 'input',
PLAY = 'play',
PAUSE = 'pause',
SECURITY_POLICY_VIOLATION = 'securitypolicyviolation',
}

export enum ResourceType {
Expand Down
70 changes: 70 additions & 0 deletions packages/core/test/specHelper.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import type { EndpointBuilder } from '../src/domain/configuration'
import type { ReportType, BrowserWindow, Report, ReportingObserver } from '../src/domain/report/browser.types'
import { instrumentMethod } from '../src/tools/instrumentMethod'
import { resetNavigationStart } from '../src/tools/timeUtils'
import { buildUrl } from '../src/tools/urlPolyfill'
Expand Down Expand Up @@ -417,3 +418,72 @@ export function stubCookie() {
},
}
}

export function stubReportingObserver() {
amortemousque marked this conversation as resolved.
Show resolved Hide resolved
const originalReportingObserver = (window as BrowserWindow).ReportingObserver
let callbacks: Array<(reports: Report[]) => void> = []

;(window as BrowserWindow).ReportingObserver = function (callback: (reports: Report[]) => void) {
callbacks.push(callback)
return {
disconnect() {
noop()
},
observe() {
noop()
},
takeRecords() {
return []
},
}
} as unknown as ReportingObserver

return {
raiseReport(type: ReportType) {
callbacks.forEach((callback) => callback([{ ...FAKE_REPORT, type }]))
},
reset() {
;(window as BrowserWindow).ReportingObserver = originalReportingObserver
callbacks = []
},
}
}

export function stubCspEventListener() {
spyOn(document, 'addEventListener').and.callFake((_type: string, listener: EventListener) => {
listeners.push(listener)
})

const listeners: EventListener[] = []

return {
dispatchEvent() {
listeners.forEach((listener) => listener(FAKE_CSP_VIOLATION_EVENT))
},
}
}

export const FAKE_CSP_VIOLATION_EVENT = {
blockedURI: 'blob',
columnNumber: 8,
documentURI: 'blob',
effectiveDirective: 'worker-src',
lineNumber: 17,
originalPolicy: "worker-src 'none'",
referrer: '',
sourceFile: 'http://foo.bar/index.js',
statusCode: 200,
violatedDirective: 'worker-src',
} as SecurityPolicyViolationEvent

export const FAKE_REPORT: Report = {
type: 'intervention',
url: 'http://foo.bar',
body: {
id: 'NavigatorVibrate',
columnNumber: 10,
lineNumber: 20,
message: 'foo bar',
sourceFile: 'http://foo.bar/index.js',
},
}