123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311 |
- import {browserHistory, createRoutes, match} from 'react-router';
- import {extraErrorDataIntegration} from '@sentry/integrations';
- import * as Sentry from '@sentry/react';
- import {_browserPerformanceTimeOriginMode} from '@sentry/utils';
- import type {Event} from '@sentry/types';
- import {SENTRY_RELEASE_VERSION, SPA_DSN} from 'sentry/constants';
- import type {Config} from 'sentry/types/system';
- import {addExtraMeasurements, addUIElementTag} from 'sentry/utils/performanceForSentry';
- import {normalizeUrl} from 'sentry/utils/withDomainRequired';
- import {getErrorDebugIds} from 'sentry/utils/getErrorDebugIds';
- const SPA_MODE_ALLOW_URLS = [
- 'localhost',
- 'dev.getsentry.net',
- 'sentry.dev',
- 'webpack-internal://',
- ];
- const SPA_MODE_TRACE_PROPAGATION_TARGETS = [
- 'localhost',
- 'dev.getsentry.net',
- 'sentry.dev',
- ];
- let lastEventId: string | undefined;
- export function getLastEventId(): string | undefined {
- return lastEventId;
- }
- const IGNORED_BREADCRUMB_FETCH_HOSTS = ['amplitude.com', 'reload.getsentry.net'];
- const IGNORED_SPANS_BY_DESCRIPTION = ['amplitude.com', 'reload.getsentry.net'];
- const shouldOverrideBrowserProfiling = window?.__initialData?.user?.isSuperuser;
- function getSentryIntegrations(routes?: Function) {
- const integrations = [
- extraErrorDataIntegration({
-
- depth: 6,
- }),
- Sentry.metrics.metricsAggregatorIntegration(),
- Sentry.reactRouterV3BrowserTracingIntegration({
- history: browserHistory as any,
- routes: typeof routes === 'function' ? createRoutes(routes()) : [],
- match,
- _experiments: {
- enableInteractions: true,
- },
- enableInp: true,
- }),
- Sentry.browserProfilingIntegration(),
- ];
- return integrations;
- }
- export function initializeSdk(config: Config, {routes}: {routes?: Function} = {}) {
- const {apmSampling, sentryConfig, userIdentity} = config;
- const tracesSampleRate = apmSampling ?? 0;
- const extraTracePropagationTargets = SPA_DSN
- ? SPA_MODE_TRACE_PROPAGATION_TARGETS
- : [...sentryConfig?.tracePropagationTargets];
- Sentry.init({
- ...sentryConfig,
-
- dsn: SPA_DSN || sentryConfig?.dsn,
-
- release: SENTRY_RELEASE_VERSION ?? sentryConfig?.release,
- allowUrls: SPA_DSN ? SPA_MODE_ALLOW_URLS : sentryConfig?.allowUrls,
- integrations: getSentryIntegrations(routes),
- tracesSampleRate,
- profilesSampleRate: shouldOverrideBrowserProfiling ? 1 : 0.1,
- tracePropagationTargets: ['localhost', /^\//, ...extraTracePropagationTargets],
- tracesSampler: context => {
- if (context.transactionContext.op?.startsWith('ui.action')) {
- return tracesSampleRate / 100;
- }
- return tracesSampleRate;
- },
- beforeSendTransaction(event) {
- addExtraMeasurements(event);
- addUIElementTag(event);
- event.spans = event.spans?.filter(span => {
- return IGNORED_SPANS_BY_DESCRIPTION.every(
- partialDesc => !span.description?.includes(partialDesc)
- );
- });
- if (event.transaction) {
- event.transaction = normalizeUrl(event.transaction, {forceCustomerDomain: true});
- }
- return event;
- },
- ignoreErrors: [
-
- 'AbortError: Fetch is aborted',
-
- "NotFoundError: Failed to execute 'removeChild' on 'Node': The node to be removed is not a child of this node.",
- "NotFoundError: Failed to execute 'insertBefore' on 'Node': The node before which the new node is to be inserted is not a child of this node.",
- ],
- beforeBreadcrumb(crumb) {
- const isFetch = crumb.category === 'fetch' || crumb.category === 'xhr';
-
- if (
- isFetch &&
- IGNORED_BREADCRUMB_FETCH_HOSTS.some(host => crumb.data?.url?.includes(host))
- ) {
- return null;
- }
- return crumb;
- },
- beforeSend(event, hint) {
- if (isFilteredRequestErrorEvent(event) || isEventWithFileUrl(event)) {
- return null;
- }
- handlePossibleUndefinedResponseBodyErrors(event);
- addEndpointTagToRequestError(event);
- lastEventId = event.event_id || hint.event_id;
- return event;
- },
- });
- if (process.env.NODE_ENV !== 'production') {
- if (sentryConfig.environment === 'development' && process.env.NO_SPOTLIGHT !== '1') {
- import('@spotlightjs/spotlight').then(Spotlight => {
- Spotlight.init();
- });
- }
- }
-
-
- const debugIdPolyfillEventProcessor = async (event: Event, hint: Sentry.EventHint) => {
- if (!(hint.originalException instanceof Error)) {
- return event;
- }
- try {
- const debugIdMap = await getErrorDebugIds(hint.originalException);
-
- event.debug_meta = {};
- event.debug_meta.images = [];
- const images = event.debug_meta.images;
- Object.keys(debugIdMap).forEach(filename => {
- images.push({
- type: 'sourcemap',
- code_file: filename,
- debug_id: debugIdMap[filename],
- });
- });
- } catch (e) {
- event.extra = event.extra || {};
- event.extra.debug_id_fetch_error = String(e);
- }
- return event;
- };
- debugIdPolyfillEventProcessor.id = 'debugIdPolyfillEventProcessor';
- Sentry.addEventProcessor(debugIdPolyfillEventProcessor);
-
- Sentry.addEventProcessor((event: Sentry.Event, _hint?: Sentry.EventHint) => {
- event.tags = event.tags || {};
- event.tags['timeOrigin.mode'] = _browserPerformanceTimeOriginMode;
- return event;
- });
- if (userIdentity) {
- Sentry.setUser(userIdentity);
- }
- if (window.__SENTRY__VERSION) {
- Sentry.setTag('sentry_version', window.__SENTRY__VERSION);
- }
- const {customerDomain} = window.__initialData;
- if (customerDomain) {
- Sentry.setTag('isCustomerDomain', 'yes');
- Sentry.setTag('customerDomain.organizationUrl', customerDomain.organizationUrl);
- Sentry.setTag('customerDomain.sentryUrl', customerDomain.sentryUrl);
- Sentry.setTag('customerDomain.subdomain', customerDomain.subdomain);
- }
- }
- export function isFilteredRequestErrorEvent(event: Event): boolean {
- const exceptionValues = event.exception?.values;
- if (!exceptionValues) {
- return false;
- }
-
-
-
-
- const mainAndMaybeCauseErrors = exceptionValues.slice(-2);
- for (const error of mainAndMaybeCauseErrors) {
- const {type = '', value = ''} = error;
- const is200 =
- ['RequestError'].includes(type) && !!value.match('(GET|POST|PUT|DELETE) .* 200');
- const is400 =
- ['BadRequestError', 'RequestError'].includes(type) &&
- !!value.match('(GET|POST|PUT|DELETE) .* 400');
- const is401 =
- ['UnauthorizedError', 'RequestError'].includes(type) &&
- !!value.match('(GET|POST|PUT|DELETE) .* 401');
- const is403 =
- ['ForbiddenError', 'RequestError'].includes(type) &&
- !!value.match('(GET|POST|PUT|DELETE) .* 403');
- const is404 =
- ['NotFoundError', 'RequestError'].includes(type) &&
- !!value.match('(GET|POST|PUT|DELETE) .* 404');
- const is429 =
- ['TooManyRequestsError', 'RequestError'].includes(type) &&
- !!value.match('(GET|POST|PUT|DELETE) .* 429');
- if (is200 || is400 || is401 || is403 || is404 || is429) {
- return true;
- }
- }
- return false;
- }
- export function isEventWithFileUrl(event: Event): boolean {
- return !!event.request?.url?.startsWith('file://');
- }
- function handlePossibleUndefinedResponseBodyErrors(event: Event): void {
-
- const [mainError, causeError] = event.exception?.values?.slice(-2).reverse() || [];
- const mainErrorIsURBE = mainError?.type === 'UndefinedResponseBodyError';
- const causeErrorIsURBE = causeError?.type === 'UndefinedResponseBodyError';
- if (mainErrorIsURBE || causeErrorIsURBE) {
- mainError.type = 'UndefinedResponseBodyError';
- event.tags = {...event.tags, undefinedResponseBody: true};
- event.fingerprint = mainErrorIsURBE
- ? ['UndefinedResponseBodyError as main error']
- : ['UndefinedResponseBodyError as cause error'];
- }
- }
- export function addEndpointTagToRequestError(event: Event): void {
- const errorMessage = event.exception?.values?.[0].value || '';
-
- const requestErrorRegex = new RegExp('^([A-Za-z]+ (/[^/]+)+/) \\d+$');
- const messageMatch = requestErrorRegex.exec(errorMessage);
- if (messageMatch) {
- event.tags = {...event.tags, endpoint: messageMatch[1]};
- }
- }
|