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(browser): Add outgoingRequest context to fetch errors #15850

Draft
wants to merge 1 commit into
base: develop
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
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
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ window.credentialsInUrl = () => {

// Invalid mode
window.invalidMode = () => {
fetch('https://sentry-test-external.io/invalid-mode', { mode: 'navigate' });
fetch('http://sentry-test-external.io/invalid-mode', { mode: 'navigate' });
};

// Invalid request method
Expand Down
36 changes: 36 additions & 0 deletions dev-packages/browser-integration-tests/suites/errors/fetch/test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,10 @@ sentryTest('handles fetch network errors @firefox', async ({ getLocalTestUrl, pa
type: 'onunhandledrejection',
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'GET',
url: 'http://sentry-test-external.io/does-not-exist',
});
});


Expand Down Expand Up @@ -55,6 +59,10 @@ sentryTest('handles fetch network errors on subdomains @firefox', async ({ getLo
type: 'onunhandledrejection',
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'GET',
url: 'http://subdomain.sentry-test-external.io/does-not-exist',
});
});

sentryTest('handles fetch invalid header name errors @firefox', async ({ getLocalTestUrl, page, browserName }) => {
Expand Down Expand Up @@ -85,6 +93,10 @@ sentryTest('handles fetch invalid header name errors @firefox', async ({ getLoca
frames: expect.any(Array),
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'GET',
url: 'http://sentry-test-external.io/invalid-header-name',
});
});

sentryTest('handles fetch invalid header value errors @firefox', async ({ getLocalTestUrl, page, browserName }) => {
Expand Down Expand Up @@ -117,6 +129,10 @@ sentryTest('handles fetch invalid header value errors @firefox', async ({ getLoc
frames: expect.any(Array),
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'GET',
url: 'http://sentry-test-external.io/invalid-header-value',
});
});

sentryTest('handles fetch invalid URL scheme errors @firefox', async ({ getLocalTestUrl, page, browserName }) => {
Expand Down Expand Up @@ -159,6 +175,10 @@ sentryTest('handles fetch invalid URL scheme errors @firefox', async ({ getLocal
frames: expect.any(Array),
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'GET',
url: 'blub://sentry-test-external.io/invalid-scheme',
});
});

sentryTest('handles fetch credentials in url errors @firefox', async ({ getLocalTestUrl, page, browserName }) => {
Expand Down Expand Up @@ -191,6 +211,10 @@ sentryTest('handles fetch credentials in url errors @firefox', async ({ getLocal
frames: expect.any(Array),
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'GET',
url: 'https://user:password@sentry-test-external.io/credentials-in-url',
});
});

sentryTest('handles fetch invalid mode errors @firefox', async ({ getLocalTestUrl, page, browserName }) => {
Expand Down Expand Up @@ -222,6 +246,10 @@ sentryTest('handles fetch invalid mode errors @firefox', async ({ getLocalTestUr
frames: expect.any(Array),
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'GET',
url: 'http://sentry-test-external.io/invalid-mode',
});
});

sentryTest('handles fetch invalid request method errors @firefox', async ({ getLocalTestUrl, page, browserName }) => {
Expand Down Expand Up @@ -252,6 +280,10 @@ sentryTest('handles fetch invalid request method errors @firefox', async ({ getL
frames: expect.any(Array),
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'CONNECT',
url: 'http://sentry-test-external.io/invalid-method',
});
});

sentryTest(
Expand Down Expand Up @@ -284,5 +316,9 @@ sentryTest(
frames: expect.any(Array),
},
});
expect(eventData.contexts?.outgoingRequest).toEqual({
method: 'PUT',
url: 'http://sentry-test-external.io/no-cors-method',
});
},
);
11 changes: 11 additions & 0 deletions packages/core/src/utils-hoist/instrument/fetch.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import { Scope } from '../../scope';
import type { HandlerDataFetch } from '../../types-hoist';
import { addScopeDataToError } from '../../utils/prepareEvent';

import { isError } from '../is';
import { addNonEnumerableProperty, fill } from '../object';
Expand Down Expand Up @@ -126,6 +128,15 @@ function instrumentFetch(onFetchResolved?: (response: Response) => void, skipNat
}
}

// We attach an additional scope to the error, which contains the outgoing request data
// If this error bubbles up and is captured by Sentry, the scope will be added to the event
const scope = new Scope();
scope.setContext('outgoingRequest', {
method,
url,
});
addScopeDataToError(error, scope);

// NOTE: If you are a Sentry user, and you are seeing this stack frame,
// it means the sentry.javascript SDK caught an error invoking your application code.
// This is expected behavior and NOT indicative of a bug with sentry.javascript.
Expand Down
23 changes: 23 additions & 0 deletions packages/core/src/utils/prepareEvent.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,19 @@ export type ExclusiveEventHintOrCaptureContext =
| (CaptureContext & Partial<{ [key in keyof EventHint]: never }>)
| (EventHint & Partial<{ [key in keyof ScopeContext]: never }>);

const errorScopeMap = new WeakMap<Error, Scope>();

/**
* Add a scope that should be applied to the given error, if it is captured by Sentry.
*/
export function addScopeDataToError(error: Error, scope: Scope): void {
try {
errorScopeMap.set(error, scope);
} catch {
// ignore it if errors happen here, e.g. if `error` is not an object
}
}

/**
* Adds common information to events.
*
Expand Down Expand Up @@ -84,6 +97,16 @@ export function prepareEvent(
mergeScopeData(data, isolationData);
}

// In some cases, additional scope data may be attached to an error
// We also merge this data into the event scope data, if available
const originalException = hint.originalException;
if (originalException instanceof Error) {
const additionalErrorScope = errorScopeMap.get(originalException);
if (additionalErrorScope) {
mergeScopeData(data, additionalErrorScope.getScopeData());
}
}

if (finalScope) {
const finalScopeData = finalScope.getScopeData();
mergeScopeData(data, finalScopeData);
Expand Down
Loading