Skip to content

ref(browser): Refactor browser integrations to avoid setupOnce #9898

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

Merged
merged 1 commit into from
Dec 19, 2023
Merged
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
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;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we can just remove this tbh, nobody should be relying on it.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yeah, I'll remove it in a next step where I rewrite this to functional style!


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