Skip to content

ref(core)!: Remove backwards compatible SentryCarrier type #14697

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 6 commits into from
Dec 16, 2024
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
38 changes: 30 additions & 8 deletions packages/core/src/carrier.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import type { AsyncContextStack } from './asyncContext/stackStrategy';
import type { AsyncContextStrategy } from './asyncContext/types';
import type { Client, Integration, MetricsAggregator, Scope } from './types-hoist';
import type { Client, MetricsAggregator, Scope } from './types-hoist';
import type { Logger } from './utils-hoist/logger';
import { SDK_VERSION } from './utils-hoist/version';
import { GLOBAL_OBJ } from './utils-hoist/worldwide';

Expand All @@ -16,21 +17,20 @@ type VersionedCarrier = {
version?: string;
} & Record<Exclude<string, 'version'>, SentryCarrier>;

interface SentryCarrier {
export interface SentryCarrier {
acs?: AsyncContextStrategy;
stack?: AsyncContextStack;

globalScope?: Scope;
defaultIsolationScope?: Scope;
defaultCurrentScope?: Scope;
globalMetricsAggregators?: WeakMap<Client, MetricsAggregator> | undefined;
logger?: Logger;

// TODO(v9): Remove these properties - they are no longer used and were left over in v8
integrations?: Integration[];
extensions?: {
// eslint-disable-next-line @typescript-eslint/ban-types
[key: string]: Function;
};
/** Overwrites TextEncoder used in `@sentry/core`, need for `[email protected]` and older */
encodePolyfill?: (input: string) => Uint8Array;
/** Overwrites TextDecoder used in `@sentry/core`, need for `[email protected]` and older */
decodePolyfill?: (input: Uint8Array) => string;
}

/**
Expand All @@ -57,3 +57,25 @@ export function getSentryCarrier(carrier: Carrier): SentryCarrier {
// rather than what's set in .version so that "this" SDK always gets its carrier
return (__SENTRY__[SDK_VERSION] = __SENTRY__[SDK_VERSION] || {});
}

/**
* Returns a global singleton contained in the global `__SENTRY__[]` object.
*
* If the singleton doesn't already exist in `__SENTRY__`, it will be created using the given factory
* function and added to the `__SENTRY__` object.
*
* @param name name of the global singleton on __SENTRY__
* @param creator creator Factory function to create the singleton if it doesn't already exist on `__SENTRY__`
* @param obj (Optional) The global object on which to look for `__SENTRY__`, if not `GLOBAL_OBJ`'s return value
* @returns the singleton
*/
export function getGlobalSingleton<Prop extends keyof SentryCarrier>(
name: Prop,
creator: () => NonNullable<SentryCarrier[Prop]>,
obj = GLOBAL_OBJ,
): NonNullable<SentryCarrier[Prop]> {
const __SENTRY__ = (obj.__SENTRY__ = obj.__SENTRY__ || {});
const carrier = (__SENTRY__[SDK_VERSION] = __SENTRY__[SDK_VERSION] || {});
Comment on lines +77 to +78
Copy link
Member

Choose a reason for hiding this comment

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

l: we could call getSentryCarrier(obj) here to save some bytes, no?

Copy link
Member Author

Choose a reason for hiding this comment

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

actually no 😬 I had that first but it broke loader tests. This is why I added the comment below - if we'd call that, it would already set the version, which would lead to stuff thinking the SDK is set up, when it isn't. We store e.g. the default global/isolation scope on there even before the SDK is setup, so we need to ensure the version is not set here yet!

Copy link
Member Author

Choose a reason for hiding this comment

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

(on the plus side, the loader tests are good! xD)

Copy link
Member

Choose a reason for hiding this comment

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

ah sorry, I should have read the comment 🤦 thanks!

// Note: We do not want to set `carrier.version` here, as this may be called before any `init` is called, e.g. for the default scopes
return carrier[name] || (carrier[name] = creator());
}
2 changes: 1 addition & 1 deletion packages/core/src/currentScopes.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
import { getAsyncContextStrategy } from './asyncContext';
import { getMainCarrier } from './carrier';
import { getGlobalSingleton } from './carrier';
import { Scope as ScopeClass } from './scope';
import type { Client, Scope, TraceContext } from './types-hoist';
import { dropUndefinedKeys } from './utils-hoist/object';
import { getGlobalSingleton } from './utils-hoist/worldwide';

/**
* Get the currently active scope.
Expand Down
2 changes: 1 addition & 1 deletion packages/core/src/defaultScopes.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { getGlobalSingleton } from './carrier';
import { Scope as ScopeClass } from './scope';
import type { Scope } from './types-hoist';
import { getGlobalSingleton } from './utils-hoist/worldwide';

/** Get the default current scope. */
export function getDefaultCurrentScope(): Scope {
Expand Down
2 changes: 1 addition & 1 deletion packages/core/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,7 @@ export {
getDefaultIsolationScope,
} from './defaultScopes';
export { setAsyncContextStrategy } from './asyncContext';
export { getMainCarrier } from './carrier';
export { getGlobalSingleton, getMainCarrier } from './carrier';
export { makeSession, closeSession, updateSession } from './session';
// eslint-disable-next-line deprecation/deprecation
export { SessionFlusher } from './sessionflusher';
Expand Down
6 changes: 3 additions & 3 deletions packages/core/src/metrics/exports.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
import { getGlobalSingleton } from '../carrier';
import { getClient } from '../currentScopes';
import { DEBUG_BUILD } from '../debug-build';
import { startSpanManual } from '../tracing';
import type { Client, DurationUnit, MetricData, MetricsAggregator as MetricsAggregatorInterface } from '../types-hoist';
import { logger } from '../utils-hoist/logger';
import { timestampInSeconds } from '../utils-hoist/time';
import { getGlobalSingleton } from '../utils-hoist/worldwide';
import { handleCallbackErrors } from '../utils/handleCallbackErrors';
import { getActiveSpan, getRootSpan, spanToJSON } from '../utils/spanUtils';
import { COUNTER_METRIC_TYPE, DISTRIBUTION_METRIC_TYPE, GAUGE_METRIC_TYPE, SET_METRIC_TYPE } from './constants';
Expand All @@ -23,9 +23,9 @@ function getMetricsAggregatorForClient(
client: Client,
Aggregator: MetricsAggregatorConstructor,
): MetricsAggregatorInterface {
const globalMetricsAggregators = getGlobalSingleton<WeakMap<Client, MetricsAggregatorInterface>>(
const globalMetricsAggregators = getGlobalSingleton(
'globalMetricsAggregators',
() => new WeakMap(),
() => new WeakMap<Client, MetricsAggregatorInterface>(),
);

const aggregator = globalMetricsAggregators.get(client);
Expand Down
11 changes: 5 additions & 6 deletions packages/core/src/utils-hoist/envelope.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { getSentryCarrier } from '../carrier';
import type {
Attachment,
AttachmentItem,
Expand Down Expand Up @@ -74,18 +75,16 @@ export function envelopeContainsItemType(envelope: Envelope, types: EnvelopeItem
* Encode a string to UTF8 array.
*/
function encodeUTF8(input: string): Uint8Array {
return GLOBAL_OBJ.__SENTRY__ && GLOBAL_OBJ.__SENTRY__.encodePolyfill
? GLOBAL_OBJ.__SENTRY__.encodePolyfill(input)
: new TextEncoder().encode(input);
const carrier = getSentryCarrier(GLOBAL_OBJ);
return carrier.encodePolyfill ? carrier.encodePolyfill(input) : new TextEncoder().encode(input);
}

/**
* Decode a UTF8 array to string.
*/
function decodeUTF8(input: Uint8Array): string {
return GLOBAL_OBJ.__SENTRY__ && GLOBAL_OBJ.__SENTRY__.decodePolyfill
? GLOBAL_OBJ.__SENTRY__.decodePolyfill(input)
: new TextDecoder().decode(input);
const carrier = getSentryCarrier(GLOBAL_OBJ);
return carrier.decodePolyfill ? carrier.decodePolyfill(input) : new TextDecoder().decode(input);
}

/**
Expand Down
2 changes: 1 addition & 1 deletion packages/core/src/utils-hoist/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ export { getBreadcrumbLogLevelFromHttpStatusCode } from './breadcrumb-log-level'
export { getComponentName, getDomElement, getLocationHref, htmlTreeAsString } from './browser';
export { dsnFromString, dsnToString, makeDsn } from './dsn';
export { SentryError } from './error';
export { GLOBAL_OBJ, getGlobalSingleton } from './worldwide';
export { GLOBAL_OBJ } from './worldwide';
export type { InternalGlobal } from './worldwide';
export { addConsoleInstrumentationHandler } from './instrument/console';
export { addFetchEndInstrumentationHandler, addFetchInstrumentationHandler } from './instrument/fetch';
Expand Down
6 changes: 3 additions & 3 deletions packages/core/src/utils-hoist/logger.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { getGlobalSingleton } from '../carrier';
import type { ConsoleLevel } from '../types-hoist';

import { DEBUG_BUILD } from './debug-build';
import { GLOBAL_OBJ, getGlobalSingleton } from './worldwide';
import { GLOBAL_OBJ } from './worldwide';

/** Prefix for logging strings */
const PREFIX = 'Sentry Logger ';
Expand All @@ -24,7 +24,7 @@ export const originalConsoleMethods: {
[key in ConsoleLevel]?: (...args: unknown[]) => void;
} = {};

/** JSDoc */
/** A Sentry Logger instance. */
export interface Logger extends LoggerConsoleMethods {
disable(): void;
enable(): void;
Expand Down
57 changes: 2 additions & 55 deletions packages/core/src/utils-hoist/worldwide.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,40 +12,8 @@

/* eslint-disable @typescript-eslint/no-explicit-any */

import type { Client, MetricsAggregator, Scope } from '../types-hoist';

import type { Carrier } from '../carrier';
import type { SdkSource } from './env';
import type { logger } from './logger';
import { SDK_VERSION } from './version';

interface SentryCarrier {
Copy link
Member Author

Choose a reason for hiding this comment

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

this type was basically duplicated here - this comes from the days of core vs utils separation, I believe 😬

acs?: any;
stack?: any;

globalScope?: Scope;
defaultIsolationScope?: Scope;
defaultCurrentScope?: Scope;
globalMetricsAggregators?: WeakMap<Client, MetricsAggregator> | undefined;
logger?: typeof logger;

/** Overwrites TextEncoder used in `@sentry/core`, need for `[email protected]` and older */
encodePolyfill?: (input: string) => Uint8Array;
/** Overwrites TextDecoder used in `@sentry/core`, need for `[email protected]` and older */
decodePolyfill?: (input: Uint8Array) => string;
}

// TODO(v9): Clean up or remove this type
type BackwardsCompatibleSentryCarrier = SentryCarrier & {
// pre-v7 hub (replaced by .stack)
hub: any;
integrations?: any[];
logger: any;
extensions?: {
/** Extension methods for the hub, which are bound to the current Hub instance */
// eslint-disable-next-line @typescript-eslint/ban-types
[key: string]: Function;
};
};

/** Internal global with common properties and Sentry extensions */
export type InternalGlobal = {
Expand Down Expand Up @@ -73,35 +41,14 @@ export type InternalGlobal = {
* file.
*/
_sentryDebugIds?: Record<string, string>;
__SENTRY__: Record<Exclude<string, 'version'>, SentryCarrier> & {
version?: string;
} & BackwardsCompatibleSentryCarrier;
/**
* Raw module metadata that is injected by bundler plugins.
*
* Keys are `error.stack` strings, values are the metadata.
*/
_sentryModuleMetadata?: Record<string, any>;
_sentryEsmLoaderHookRegistered?: boolean;
};
} & Carrier;

/** Get's the global object for the current JavaScript runtime */
export const GLOBAL_OBJ = globalThis as unknown as InternalGlobal;

/**
* Returns a global singleton contained in the global `__SENTRY__[]` object.
*
* If the singleton doesn't already exist in `__SENTRY__`, it will be created using the given factory
* function and added to the `__SENTRY__` object.
*
* @param name name of the global singleton on __SENTRY__
* @param creator creator Factory function to create the singleton if it doesn't already exist on `__SENTRY__`
* @param obj (Optional) The global object on which to look for `__SENTRY__`, if not `GLOBAL_OBJ`'s return value
* @returns the singleton
*/
export function getGlobalSingleton<T>(name: keyof SentryCarrier, creator: () => T, obj?: unknown): T {
const gbl = (obj || GLOBAL_OBJ) as InternalGlobal;
const __SENTRY__ = (gbl.__SENTRY__ = gbl.__SENTRY__ || {});
const versionedCarrier = (__SENTRY__[SDK_VERSION] = __SENTRY__[SDK_VERSION] || {});
return versionedCarrier[name] || (versionedCarrier[name] = creator());
}
1 change: 0 additions & 1 deletion packages/core/test/lib/hint.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,6 @@ describe('Hint', () => {

afterEach(() => {
jest.clearAllMocks();
// @ts-expect-error for testing
delete GLOBAL_OBJ.__SENTRY__;
});

Expand Down
12 changes: 7 additions & 5 deletions packages/core/test/utils-hoist/envelope.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import {
spanToJSON,
} from '@sentry/core';
import { SentrySpan } from '@sentry/core';
import { getSentryCarrier } from '../../src/carrier';
import {
addItemToEnvelope,
createEnvelope,
Expand Down Expand Up @@ -107,17 +108,18 @@ describe('envelope', () => {
{
name: 'with TextEncoder/Decoder polyfill',
before: () => {
GLOBAL_OBJ.__SENTRY__ = {} as InternalGlobal['__SENTRY__'];
GLOBAL_OBJ.__SENTRY__.encodePolyfill = jest.fn<Uint8Array, [string]>((input: string) =>
GLOBAL_OBJ.__SENTRY__ = {};

getSentryCarrier(GLOBAL_OBJ).encodePolyfill = jest.fn<Uint8Array, [string]>((input: string) =>
new TextEncoder().encode(input),
);
GLOBAL_OBJ.__SENTRY__.decodePolyfill = jest.fn<string, [Uint8Array]>((input: Uint8Array) =>
getSentryCarrier(GLOBAL_OBJ).decodePolyfill = jest.fn<string, [Uint8Array]>((input: Uint8Array) =>
new TextDecoder().decode(input),
);
},
after: () => {
expect(GLOBAL_OBJ.__SENTRY__.encodePolyfill).toHaveBeenCalled();
expect(GLOBAL_OBJ.__SENTRY__.decodePolyfill).toHaveBeenCalled();
expect(getSentryCarrier(GLOBAL_OBJ).encodePolyfill).toHaveBeenCalled();
expect(getSentryCarrier(GLOBAL_OBJ).decodePolyfill).toHaveBeenCalled();
},
},
{
Expand Down
Loading