Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
3324a90
Merge pull request #9853 from getsentry/master
github-actions[bot] Dec 14, 2023
383c816
build: Enable codecov AI PR review (#9850)
AbhiPrasad Dec 14, 2023
1e8d2b3
ref: Use `getCurrentScope()`/`hub.getScope()` instead of `configureSc…
mydea Dec 15, 2023
da8c9c3
feat(nextjs): Auto instrument generation functions (#9781)
Dec 15, 2023
30bb8b5
fix(utils): Do not use `Event` type in worldwide (#9864)
mydea Dec 15, 2023
ffc4181
ref: Use `getCurrentScope()` / `getClient` instead of `hub.xxx` (#9862)
mydea Dec 15, 2023
5bc9a38
fix(utils): Update `eventFromUnknownInput` to avoid scope pollution &…
mydea Dec 15, 2023
72c3488
ref(node): Refactor node integrations to use `processEvent` (#9018)
mydea Dec 15, 2023
e128936
fix(utils): Support crypto.getRandomValues in old Chromium versions (…
jghinestrosa Dec 15, 2023
53724c5
meta(feedback): Fix syntax error in README (#9875)
billyvg Dec 15, 2023
12c146b
fix(nextjs): Export `createReduxEnhancer` (#9854)
adam187 Dec 18, 2023
db4bef1
ref: Pass client instead of hub to `isSentryRequestUrl` (#9869)
mydea Dec 18, 2023
f2a4caa
feat(core): Update `withScope` to return callback return value (#9866)
mydea Dec 18, 2023
01a4cc9
feat(core): Add type & utility for function-based integrations (#9818)
mydea Dec 18, 2023
c0c5eca
feat(node): Add Hapi Integration (#9539)
onurtemizkan Dec 18, 2023
35906d0
ref: Use `addBreadcrumb` directly & allow to pass hint (#9867)
mydea Dec 18, 2023
f922414
build: Fix linting (#9888)
mydea Dec 18, 2023
e1d3633
ref(serverless): Avoid using `pushScope` (#9883)
mydea Dec 18, 2023
ada32b2
ref(nextjs): Simplify `wrapServerComponentWithSentry` (#9844)
Dec 18, 2023
0efdb21
feat(core): Deprecate `configureScope` (#9887)
mydea Dec 18, 2023
b27c236
feat(core): Deprecate `pushScope` & `popScope` (#9890)
mydea Dec 18, 2023
f121585
feat(nextjs): Connect server component transactions if there is no in…
Dec 18, 2023
8fb1a2f
feat(deno): Support `Deno.CronSchedule` for cron jobs (#9880)
timfish Dec 18, 2023
84299d0
feat(replay): Add `canvas.type` setting (#9877)
billyvg Dec 18, 2023
605fd50
feat(node-experimental): Update to new Scope APIs (#9799)
mydea Dec 19, 2023
32f4bf0
ref(deno): Refactor deno integration to avoid `setupOnce` (#9900)
mydea Dec 19, 2023
91a6b4e
ref(browser): Refactor browser integrations to avoid `setupOnce` (#9898)
mydea Dec 19, 2023
c9552a4
ref(node): Refactor LocalVariables integration to avoid `setupOnce` (…
mydea Dec 19, 2023
4e0c460
chore(sveltekit): Add SvelteKit 2.0 to peer dependencies (#9861)
Lms24 Dec 19, 2023
6d32228
fix(sveltekit): Add conditional exports (#9872)
Lms24 Dec 19, 2023
a856913
fix(remix): Do not capture thrown redirect responses. (#9909)
onurtemizkan Dec 19, 2023
31c769c
test(sveltekit): Add SvelteKit 2.0 E2E test app (#9873)
Lms24 Dec 19, 2023
cef3621
ref(sveltekit): Improve SvelteKit 2.0 404 server error handling (#9901)
Lms24 Dec 19, 2023
cf773fc
fix(sveltekit): Avoid capturing 404 errors on client side (#9902)
Lms24 Dec 19, 2023
9f173e1
meta(changelog): Update changelog for 7.89.0
Dec 19, 2023
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
Prev Previous commit
Next Next commit
ref(browser): Refactor browser integrations to avoid setupOnce (#9898)
Also a small core refactor...
  • Loading branch information
mydea authored Dec 19, 2023
commit 91a6b4e5ae4d5cfe745b548a20ecefad8f4d3f0a
73 changes: 27 additions & 46 deletions packages/browser/src/integrations/globalhandlers.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
/* eslint-disable @typescript-eslint/no-unsafe-member-access */
import { getCurrentHub } from '@sentry/core';
import type { Event, Hub, Integration, Primitive, StackParser } from '@sentry/types';
import { captureEvent, getClient } from '@sentry/core';
import type { Client, Event, Integration, Primitive, StackParser } from '@sentry/types';
import {
addGlobalErrorInstrumentationHandler,
addGlobalUnhandledRejectionInstrumentationHandler,
Expand Down Expand Up @@ -36,12 +36,6 @@ export class GlobalHandlers implements Integration {
/** JSDoc */
private readonly _options: GlobalHandlersIntegrations;

/**
* Stores references functions to installing handlers. Will set to undefined
* after they have been run so that they are not used twice.
*/
private _installFunc: Record<GlobalHandlersIntegrationsOptionKeys, (() => void) | undefined>;

/** JSDoc */
public constructor(options?: GlobalHandlersIntegrations) {
this.name = GlobalHandlers.id;
Expand All @@ -50,43 +44,36 @@ export class GlobalHandlers implements Integration {
onunhandledrejection: true,
...options,
};

this._installFunc = {
onerror: _installGlobalOnErrorHandler,
onunhandledrejection: _installGlobalOnUnhandledRejectionHandler,
};
}
/**
* @inheritDoc
*/
public setupOnce(): void {
Error.stackTraceLimit = 50;
const options = this._options;

// We can disable guard-for-in as we construct the options object above + do checks against
// `this._installFunc` for the property.
// eslint-disable-next-line guard-for-in
for (const key in options) {
const installFunc = this._installFunc[key as GlobalHandlersIntegrationsOptionKeys];
if (installFunc && options[key as GlobalHandlersIntegrationsOptionKeys]) {
globalHandlerLog(key);
installFunc();
this._installFunc[key as GlobalHandlersIntegrationsOptionKeys] = undefined;
}
}

/** @inheritdoc */
public setup(client: Client): void {
if (this._options.onerror) {
_installGlobalOnErrorHandler(client);
globalHandlerLog('onerror');
}
if (this._options.onunhandledrejection) {
_installGlobalOnUnhandledRejectionHandler(client);
globalHandlerLog('onunhandledrejection');
}
}
}

function _installGlobalOnErrorHandler(): void {
function _installGlobalOnErrorHandler(client: Client): void {
addGlobalErrorInstrumentationHandler(data => {
const [hub, stackParser, attachStacktrace] = getHubAndOptions();
if (!hub.getIntegration(GlobalHandlers)) {
const { stackParser, attachStacktrace } = getOptions();

if (getClient() !== client || shouldIgnoreOnError()) {
return;
}

const { msg, url, line, column, error } = data;
if (shouldIgnoreOnError()) {
return;
}

const event =
error === undefined && isString(msg)
Expand All @@ -100,7 +87,7 @@ function _installGlobalOnErrorHandler(): void {

event.level = 'error';

hub.captureEvent(event, {
captureEvent(event, {
originalException: error,
mechanism: {
handled: false,
Expand All @@ -110,15 +97,12 @@ function _installGlobalOnErrorHandler(): void {
});
}

function _installGlobalOnUnhandledRejectionHandler(): void {
function _installGlobalOnUnhandledRejectionHandler(client: Client): void {
addGlobalUnhandledRejectionInstrumentationHandler(e => {
const [hub, stackParser, attachStacktrace] = getHubAndOptions();
if (!hub.getIntegration(GlobalHandlers)) {
return;
}
const { stackParser, attachStacktrace } = getOptions();

if (shouldIgnoreOnError()) {
return true;
if (getClient() !== client || shouldIgnoreOnError()) {
return;
}

const error = _getUnhandledRejectionError(e as unknown);
Expand All @@ -129,15 +113,13 @@ function _installGlobalOnUnhandledRejectionHandler(): void {

event.level = 'error';

hub.captureEvent(event, {
captureEvent(event, {
originalException: error,
mechanism: {
handled: false,
type: 'onunhandledrejection',
},
});

return;
});
}

Expand Down Expand Up @@ -258,12 +240,11 @@ function globalHandlerLog(type: string): void {
DEBUG_BUILD && logger.log(`Global Handler attached: ${type}`);
}

function getHubAndOptions(): [Hub, StackParser, boolean | undefined] {
const hub = getCurrentHub();
const client = hub.getClient<BrowserClient>();
function getOptions(): { stackParser: StackParser; attachStacktrace?: boolean } {
const client = getClient<BrowserClient>();
const options = (client && client.getOptions()) || {
stackParser: () => [],
attachStacktrace: false,
};
return [hub, options.stackParser, options.attachStacktrace];
return options;
}
124 changes: 64 additions & 60 deletions packages/browser/src/profiling/integration.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import type { EventEnvelope, EventProcessor, Hub, Integration, Transaction } from '@sentry/types';
import { getCurrentScope } from '@sentry/core';
import type { Client, EventEnvelope, EventProcessor, Hub, Integration, Transaction } from '@sentry/types';
import type { Profile } from '@sentry/types/src/profiling';
import { logger } from '@sentry/utils';

Expand Down Expand Up @@ -29,6 +30,7 @@ export class BrowserProfilingIntegration implements Integration {

public readonly name: string;

/** @deprecated This is never set. */
public getCurrentHub?: () => Hub;

public constructor() {
Expand All @@ -38,12 +40,13 @@ export class BrowserProfilingIntegration implements Integration {
/**
* @inheritDoc
*/
public setupOnce(_addGlobalEventProcessor: (callback: EventProcessor) => void, getCurrentHub: () => Hub): void {
this.getCurrentHub = getCurrentHub;
public setupOnce(_addGlobalEventProcessor: (callback: EventProcessor) => void, _getCurrentHub: () => Hub): void {
// noop
}

const hub = this.getCurrentHub();
const client = hub.getClient();
const scope = hub.getScope();
/** @inheritdoc */
public setup(client: Client): void {
const scope = getCurrentScope();

const transaction = scope.getTransaction();

Expand All @@ -53,67 +56,68 @@ export class BrowserProfilingIntegration implements Integration {
}
}

if (client && typeof client.on === 'function') {
client.on('startTransaction', (transaction: Transaction) => {
if (shouldProfileTransaction(transaction)) {
startProfileForTransaction(transaction);
if (typeof client.on !== 'function') {
logger.warn('[Profiling] Client does not support hooks, profiling will be disabled');
return;
}

client.on('startTransaction', (transaction: Transaction) => {
if (shouldProfileTransaction(transaction)) {
startProfileForTransaction(transaction);
}
});

client.on('beforeEnvelope', (envelope): void => {
// if not profiles are in queue, there is nothing to add to the envelope.
if (!getActiveProfilesCount()) {
return;
}

const profiledTransactionEvents = findProfiledTransactionsFromEnvelope(envelope);
if (!profiledTransactionEvents.length) {
return;
}

const profilesToAddToEnvelope: Profile[] = [];

for (const profiledTransaction of profiledTransactionEvents) {
const context = profiledTransaction && profiledTransaction.contexts;
const profile_id = context && context['profile'] && context['profile']['profile_id'];
const start_timestamp = context && context['profile'] && context['profile']['start_timestamp'];

if (typeof profile_id !== 'string') {
DEBUG_BUILD && logger.log('[Profiling] cannot find profile for a transaction without a profile context');
continue;
}
});

client.on('beforeEnvelope', (envelope): void => {
// if not profiles are in queue, there is nothing to add to the envelope.
if (!getActiveProfilesCount()) {
return;
if (!profile_id) {
DEBUG_BUILD && logger.log('[Profiling] cannot find profile for a transaction without a profile context');
continue;
}

const profiledTransactionEvents = findProfiledTransactionsFromEnvelope(envelope);
if (!profiledTransactionEvents.length) {
return;
// Remove the profile from the transaction context before sending, relay will take care of the rest.
if (context && context['profile']) {
delete context.profile;
}

const profilesToAddToEnvelope: Profile[] = [];

for (const profiledTransaction of profiledTransactionEvents) {
const context = profiledTransaction && profiledTransaction.contexts;
const profile_id = context && context['profile'] && context['profile']['profile_id'];
const start_timestamp = context && context['profile'] && context['profile']['start_timestamp'];

if (typeof profile_id !== 'string') {
DEBUG_BUILD && logger.log('[Profiling] cannot find profile for a transaction without a profile context');
continue;
}

if (!profile_id) {
DEBUG_BUILD && logger.log('[Profiling] cannot find profile for a transaction without a profile context');
continue;
}

// Remove the profile from the transaction context before sending, relay will take care of the rest.
if (context && context['profile']) {
delete context.profile;
}

const profile = takeProfileFromGlobalCache(profile_id);
if (!profile) {
DEBUG_BUILD && logger.log(`[Profiling] Could not retrieve profile for transaction: ${profile_id}`);
continue;
}

const profileEvent = createProfilingEvent(
profile_id,
start_timestamp as number | undefined,
profile,
profiledTransaction as ProfiledEvent,
);
if (profileEvent) {
profilesToAddToEnvelope.push(profileEvent);
}
const profile = takeProfileFromGlobalCache(profile_id);
if (!profile) {
DEBUG_BUILD && logger.log(`[Profiling] Could not retrieve profile for transaction: ${profile_id}`);
continue;
}

addProfilesToEnvelope(envelope as EventEnvelope, profilesToAddToEnvelope);
});
} else {
logger.warn('[Profiling] Client does not support hooks, profiling will be disabled');
}
const profileEvent = createProfilingEvent(
profile_id,
start_timestamp as number | undefined,
profile,
profiledTransaction as ProfiledEvent,
);
if (profileEvent) {
profilesToAddToEnvelope.push(profileEvent);
}
}

addProfilesToEnvelope(envelope as EventEnvelope, profilesToAddToEnvelope);
});
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import {
addTracingExtensions,
captureException,
continueTrace,
getCurrentHub,
getClient,
getCurrentScope,
runWithAsyncContext,
trace,
Expand Down Expand Up @@ -34,7 +34,7 @@ export function wrapGenerationFunctionWithSentry<F extends (...args: any[]) => a
}

let data: Record<string, unknown> | undefined = undefined;
if (getCurrentHub().getClient()?.getOptions().sendDefaultPii) {
if (getClient()?.getOptions().sendDefaultPii) {
const props: unknown = args[0];
const params = props && typeof props === 'object' && 'params' in props ? props.params : undefined;
const searchParams =
Expand Down