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-1176] collect other console logs new #1316

Merged
merged 30 commits into from
Feb 16, 2022
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
e552db1
Add console observable
amortemousque Feb 4, 2022
bf39e51
🔥 Delete trackConsoleError
amortemousque Feb 4, 2022
4ba65d7
Add console observable to rum
amortemousque Feb 4, 2022
3ef80e6
Add console observable to Logs
amortemousque Feb 4, 2022
a1c5cbb
👌 Add trackConsoleError
amortemousque Feb 7, 2022
933269a
👌 Update console log event observable
amortemousque Feb 7, 2022
eed7e68
👌 Move source computation to Log and RUM
amortemousque Feb 7, 2022
e6a6b43
Use an enum + keep current behavior
amortemousque Feb 7, 2022
03a9de3
Add feature flag forward-logs
amortemousque Feb 7, 2022
60ea273
✅ Add tests
amortemousque Feb 8, 2022
7ff6239
✨ Add forwardConsoleLogs init config option
amortemousque Feb 8, 2022
1063bad
Fix rum console error tracking
amortemousque Feb 8, 2022
cf3a4cd
Merge branch 'main' into aymeric/collect-other-console-logs-new
amortemousque Feb 8, 2022
7fc700c
✅ Add test for forwardErrorsToLogs: 'all'
amortemousque Feb 8, 2022
4abf4f2
👌 Namings
amortemousque Feb 9, 2022
9a874c3
👌 Ensure that only allowed api are instrumented.
amortemousque Feb 9, 2022
32a4393
👌 Remove CONSOLE_APIS
amortemousque Feb 9, 2022
4b8ea23
👌 Clearer cast comment
amortemousque Feb 9, 2022
197dadc
Update error observable naming
amortemousque Feb 9, 2022
d10b0dd
👌 Move console observable into domain
amortemousque Feb 9, 2022
5cd4d91
👌 Add mergeObservables tests
amortemousque Feb 9, 2022
18b2dbe
✅ Update test to ensure the api is instrumented once
amortemousque Feb 9, 2022
f88969c
Make removeDuplicates IE compatible
amortemousque Feb 9, 2022
aa2669c
Merge branch 'main' into aymeric/collect-other-console-logs-new
amortemousque Feb 10, 2022
d3792a5
👌 Code style
amortemousque Feb 10, 2022
3c3176c
👌 Add removeDuplicates test
amortemousque Feb 10, 2022
fc3d57e
Fix naming
amortemousque Feb 10, 2022
7e26428
Fix test flackyness
amortemousque Feb 11, 2022
e35baea
Merge branch 'main' into aymeric/collect-other-console-logs-new
amortemousque Feb 15, 2022
109ce1f
Simplify validateAndBuildLogsConfiguration
amortemousque Feb 15, 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
108 changes: 108 additions & 0 deletions packages/core/src/domain/console/consoleObservable.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
/* eslint-disable no-console */
import { isIE } from '../../tools/browserDetection'
import type { Subscription } from '../../tools/observable'
import type { ConsoleLog } from './consoleObservable'
import { ConsoleApiName, initConsoleObservable } from './consoleObservable'

// prettier: avoid formatting issue
// cf https://github.com/prettier/prettier/issues/12211
;[ConsoleApiName.log, ConsoleApiName.info, ConsoleApiName.warn, ConsoleApiName.debug, ConsoleApiName.error].forEach(
(api) => {
describe(`console ${api} observable`, () => {
let consoleStub: jasmine.Spy
let consoleSubscription: Subscription
let notifyLog: jasmine.Spy

beforeEach(() => {
consoleStub = spyOn(console, api)
notifyLog = jasmine.createSpy('notifyLog')

consoleSubscription = initConsoleObservable([api]).subscribe(notifyLog)
})

afterEach(() => {
consoleSubscription.unsubscribe()
})

it(`should notify ${api}`, () => {
console[api]('foo', 'bar')

const consoleLog = notifyLog.calls.mostRecent().args[0]

expect(consoleLog).toEqual(
jasmine.objectContaining({
message: `console ${api}: foo bar`,
api,
})
)
})

it('should keep original behavior', () => {
console[api]('foo', 'bar')

expect(consoleStub).toHaveBeenCalledWith('foo', 'bar')
})

it('should format error instance', () => {
console[api](new TypeError('hello'))
const consoleLog = notifyLog.calls.mostRecent().args[0]
expect(consoleLog.message).toBe(`console ${api}: TypeError: hello`)
})

it('should stringify object parameters', () => {
console[api]('Hello', { foo: 'bar' })
const consoleLog = notifyLog.calls.mostRecent().args[0]
expect(consoleLog.message).toBe(`console ${api}: Hello {\n "foo": "bar"\n}`)
})

it('should allow multiple callers', () => {
const notifyOtherCaller = jasmine.createSpy('notifyOtherCaller')
const instrumentedConsoleApi = console[api]
const otherConsoleSubscription = initConsoleObservable([api]).subscribe(notifyOtherCaller)

console[api]('foo', 'bar')

expect(instrumentedConsoleApi).toEqual(console[api])
expect(notifyLog).toHaveBeenCalledTimes(1)
expect(notifyOtherCaller).toHaveBeenCalledTimes(1)

otherConsoleSubscription.unsubscribe()
})
})
}
)

describe('console error observable', () => {
let consoleSubscription: Subscription
let notifyLog: jasmine.Spy

beforeEach(() => {
spyOn(console, 'error').and.callFake(() => true)
notifyLog = jasmine.createSpy('notifyLog')

consoleSubscription = initConsoleObservable([ConsoleApiName.error]).subscribe(notifyLog)
})

afterEach(() => {
consoleSubscription.unsubscribe()
})

it('should generate a handling stack', () => {
function triggerError() {
console.error('foo', 'bar')
}
triggerError()
const consoleLog = notifyLog.calls.mostRecent().args[0]
expect(consoleLog.handlingStack).toMatch(/^Error:\s+at triggerError (.|\n)*$/)
})

it('should extract stack from first error', () => {
console.error(new TypeError('foo'), new TypeError('bar'))
const stack = (notifyLog.calls.mostRecent().args[0] as ConsoleLog).stack
if (!isIE()) {
expect(stack).toMatch(/^TypeError: foo\s+at/)
} else {
expect(stack).toContain('TypeError: foo')
}
})
})
82 changes: 82 additions & 0 deletions packages/core/src/domain/console/consoleObservable.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
import { callMonitored } from '../internalMonitoring'
import { computeStackTrace } from '../tracekit'
import { createHandlingStack, formatErrorMessage, toStackTraceString } from '../../tools/error'
import { mergeObservables, Observable } from '../../tools/observable'
import { find, jsonStringify } from '../../tools/utils'

export const ConsoleApiName = {
log: 'log',
debug: 'debug',
info: 'info',
warn: 'warn',
error: 'error',
} as const

export type ConsoleApiName = typeof ConsoleApiName[keyof typeof ConsoleApiName]

export interface ConsoleLog {
message: string
api: ConsoleApiName
stack?: string
handlingStack?: string
}

const consoleObservablesByApi: { [k in ConsoleApiName]?: Observable<ConsoleLog> } = {}

export function initConsoleObservable(apis: ConsoleApiName[]) {
const consoleObservables = apis.map((api) => {
if (!consoleObservablesByApi[api]) {
consoleObservablesByApi[api] = createConsoleObservable(api)
}
return consoleObservablesByApi[api]!
})

return mergeObservables<ConsoleLog>(...consoleObservables)
}

/* eslint-disable no-console */
function createConsoleObservable(api: ConsoleApiName) {
const observable = new Observable<ConsoleLog>(() => {
const originalConsoleApi = console[api]

console[api] = (...params: unknown[]) => {
originalConsoleApi.apply(console, params)
const handlingStack = createHandlingStack()

callMonitored(() => {
observable.notify(buildConsoleLog(params, api, handlingStack))
})
}

return () => {
console[api] = originalConsoleApi
}
})

return observable
}

function buildConsoleLog(params: unknown[], api: ConsoleApiName, handlingStack: string): ConsoleLog {
const log: ConsoleLog = {
message: [`console ${api}:`, ...params].map((param) => formatConsoleParameters(param)).join(' '),
api,
}

if (api === ConsoleApiName.error) {
const firstErrorParam = find(params, (param: unknown): param is Error => param instanceof Error)
log.stack = firstErrorParam ? toStackTraceString(computeStackTrace(firstErrorParam)) : undefined
log.handlingStack = handlingStack
}

return log
}

function formatConsoleParameters(param: unknown) {
if (typeof param === 'string') {
return param
}
if (param instanceof Error) {
return formatErrorMessage(computeStackTrace(param))
}
return jsonStringify(param, undefined, 2)
}
91 changes: 0 additions & 91 deletions packages/core/src/domain/error/trackConsoleError.spec.ts

This file was deleted.

70 changes: 0 additions & 70 deletions packages/core/src/domain/error/trackConsoleError.ts

This file was deleted.

2 changes: 1 addition & 1 deletion packages/core/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ export {
updateExperimentalFeatures,
resetExperimentalFeatures,
} from './domain/configuration'
export { trackConsoleError } from './domain/error/trackConsoleError'
export { trackRuntimeError } from './domain/error/trackRuntimeError'
export { computeStackTrace, StackTrace } from './domain/tracekit'
export { BuildEnv, BuildMode, defineGlobal, makePublicApi } from './boot/init'
Expand Down Expand Up @@ -56,6 +55,7 @@ export { Context, ContextArray, ContextValue } from './tools/context'
export { areCookiesAuthorized, getCookie, setCookie, deleteCookie, COOKIE_ACCESS_DELAY } from './browser/cookie'
export { initXhrObservable, XhrCompleteContext, XhrStartContext } from './browser/xhrObservable'
export { initFetchObservable, FetchCompleteContext, FetchStartContext, FetchContext } from './browser/fetchObservable'
export { initConsoleObservable, ConsoleLog, ConsoleApiName } from './domain/console/consoleObservable'
export { BoundedBuffer } from './tools/boundedBuffer'
export { catchUserErrors } from './tools/catchUserErrors'
export { createContextManager } from './tools/contextManager'
Expand Down
Loading