UNPKG

@sentry/browser

Version:
279 lines (246 loc) 7.9 kB
Object.defineProperty(exports, '__esModule', { value: true }); var utils = require('@sentry/utils'); /** * This function creates an exception from a JavaScript Error */ function exceptionFromError(stackParser, ex) { // Get the frames first since Opera can lose the stack if we touch anything else first var frames = parseStackFrames(stackParser, ex); var exception = { type: ex && ex.name, value: extractMessage(ex), }; if (frames.length) { exception.stacktrace = { frames }; } if (exception.type === undefined && exception.value === '') { exception.value = 'Unrecoverable error caught'; } return exception; } /** * @hidden */ function eventFromPlainObject( stackParser, exception, syntheticException, isUnhandledRejection, ) { var event = { exception: { values: [ { type: utils.isEvent(exception) ? exception.constructor.name : isUnhandledRejection ? 'UnhandledRejection' : 'Error', value: `Non-Error ${ isUnhandledRejection ? 'promise rejection' : 'exception' } captured with keys: ${utils.extractExceptionKeysForMessage(exception)}`, }, ], }, extra: { __serialized__: utils.normalizeToSize(exception), }, }; if (syntheticException) { var frames = parseStackFrames(stackParser, syntheticException); if (frames.length) { // event.exception.values[0] has been set above (event.exception ).values[0].stacktrace = { frames }; } } return event; } /** * @hidden */ function eventFromError(stackParser, ex) { return { exception: { values: [exceptionFromError(stackParser, ex)], }, }; } /** Parses stack frames from an error */ function parseStackFrames( stackParser, ex, ) { // Access and store the stacktrace property before doing ANYTHING // else to it because Opera is not very good at providing it // reliably in other circumstances. var stacktrace = ex.stacktrace || ex.stack || ''; var popSize = getPopSize(ex); try { return stackParser(stacktrace, popSize); } catch (e) { // no-empty } return []; } // Based on our own mapping pattern - https://github.com/getsentry/sentry/blob/9f08305e09866c8bd6d0c24f5b0aabdd7dd6c59c/src/sentry/lang/javascript/errormapping.py#L83-L108 var reactMinifiedRegexp = /Minified React error #\d+;/i; function getPopSize(ex) { if (ex) { if (typeof ex.framesToPop === 'number') { return ex.framesToPop; } if (reactMinifiedRegexp.test(ex.message)) { return 1; } } return 0; } /** * There are cases where stacktrace.message is an Event object * https://github.com/getsentry/sentry-javascript/issues/1949 * In this specific case we try to extract stacktrace.message.error.message */ function extractMessage(ex) { var message = ex && ex.message; if (!message) { return 'No error message'; } if (message.error && typeof message.error.message === 'string') { return message.error.message; } return message; } /** * Creates an {@link Event} from all inputs to `captureException` and non-primitive inputs to `captureMessage`. * @hidden */ function eventFromException( stackParser, exception, hint, attachStacktrace, ) { var syntheticException = (hint && hint.syntheticException) || undefined; var event = eventFromUnknownInput(stackParser, exception, syntheticException, attachStacktrace); utils.addExceptionMechanism(event); // defaults to { type: 'generic', handled: true } event.level = 'error'; if (hint && hint.event_id) { event.event_id = hint.event_id; } return utils.resolvedSyncPromise(event); } /** * Builds and Event from a Message * @hidden */ function eventFromMessage( stackParser, message, level = 'info', hint, attachStacktrace, ) { var syntheticException = (hint && hint.syntheticException) || undefined; var event = eventFromString(stackParser, message, syntheticException, attachStacktrace); event.level = level; if (hint && hint.event_id) { event.event_id = hint.event_id; } return utils.resolvedSyncPromise(event); } /** * @hidden */ function eventFromUnknownInput( stackParser, exception, syntheticException, attachStacktrace, isUnhandledRejection, ) { let event; if (utils.isErrorEvent(exception ) && (exception ).error) { // If it is an ErrorEvent with `error` property, extract it to get actual Error var errorEvent = exception ; return eventFromError(stackParser, errorEvent.error ); } // If it is a `DOMError` (which is a legacy API, but still supported in some browsers) then we just extract the name // and message, as it doesn't provide anything else. According to the spec, all `DOMExceptions` should also be // `Error`s, but that's not the case in IE11, so in that case we treat it the same as we do a `DOMError`. // // https://developer.mozilla.org/en-US/docs/Web/API/DOMError // https://developer.mozilla.org/en-US/docs/Web/API/DOMException // https://webidl.spec.whatwg.org/#es-DOMException-specialness if (utils.isDOMError(exception ) || utils.isDOMException(exception )) { var domException = exception ; if ('stack' in (exception )) { event = eventFromError(stackParser, exception ); } else { var name = domException.name || (utils.isDOMError(domException) ? 'DOMError' : 'DOMException'); var message = domException.message ? `${name}: ${domException.message}` : name; event = eventFromString(stackParser, message, syntheticException, attachStacktrace); utils.addExceptionTypeValue(event, message); } if ('code' in domException) { event.tags = { ...event.tags, 'DOMException.code': `${domException.code}` }; } return event; } if (utils.isError(exception)) { // we have a real Error object, do nothing return eventFromError(stackParser, exception); } if (utils.isPlainObject(exception) || utils.isEvent(exception)) { // If it's a plain object or an instance of `Event` (the built-in JS kind, not this SDK's `Event` type), serialize // it manually. This will allow us to group events based on top-level keys which is much better than creating a new // group on any key/value change. var objectException = exception ; event = eventFromPlainObject(stackParser, objectException, syntheticException, isUnhandledRejection); utils.addExceptionMechanism(event, { synthetic: true, }); return event; } // If none of previous checks were valid, then it means that it's not: // - an instance of DOMError // - an instance of DOMException // - an instance of Event // - an instance of Error // - a valid ErrorEvent (one with an error property) // - a plain Object // // So bail out and capture it as a simple message: event = eventFromString(stackParser, exception , syntheticException, attachStacktrace); utils.addExceptionTypeValue(event, `${exception}`, undefined); utils.addExceptionMechanism(event, { synthetic: true, }); return event; } /** * @hidden */ function eventFromString( stackParser, input, syntheticException, attachStacktrace, ) { var event = { message: input, }; if (attachStacktrace && syntheticException) { var frames = parseStackFrames(stackParser, syntheticException); if (frames.length) { event.exception = { values: [{ value: input, stacktrace: { frames } }], }; } } return event; } exports.eventFromError = eventFromError; exports.eventFromException = eventFromException; exports.eventFromMessage = eventFromMessage; exports.eventFromPlainObject = eventFromPlainObject; exports.eventFromString = eventFromString; exports.eventFromUnknownInput = eventFromUnknownInput; exports.exceptionFromError = exceptionFromError; exports.parseStackFrames = parseStackFrames; //# sourceMappingURL=eventbuilder.js.map