From c639edf12520d1afee79e1ff2ba18df03ce70ec3 Mon Sep 17 00:00:00 2001 From: LucasZF Date: Wed, 30 Oct 2024 18:39:53 +0000 Subject: [PATCH] Fix(V6): Implement fallback system to screens that aren't reporting on the native layer the time to display. #4042 (#4189) * merge main * fix tests / yarn fix * changelog * undoo changes to sample * fix android * NIT * NIT extra line * fix-android-lint * fix java format * pub private constructor * make it final * test * add missing dependencies changelog --- CHANGELOG.md | 4 + .../io/sentry/react/RNSentryModuleImpl.java | 9 +- .../sentry/react/RNSentryTimeToDisplay.java | 42 ++++ .../java/io/sentry/react/RNSentryModule.java | 5 + .../java/io/sentry/react/RNSentryModule.java | 5 + packages/core/ios/RNSentry.mm | 9 + packages/core/ios/RNSentryTimeToDisplay.h | 7 + packages/core/ios/RNSentryTimeToDisplay.m | 43 ++++ packages/core/src/js/NativeRNSentry.ts | 1 + .../core/src/js/tracing/reactnavigation.ts | 18 +- .../js/utils/sentryeventemitterfallback.ts | 98 ++++++++ packages/core/src/js/wrapper.ts | 9 + packages/core/test/mockWrapper.ts | 1 + .../tracing/reactnavigation.ttid.test.tsx | 14 +- .../test/utils/mockedSentryeventemitter.ts | 39 --- .../utils/mockedSentryeventemitterfallback.ts | 27 ++ .../utils/sentryeventemitterfallback.test.ts | 236 ++++++++++++++++++ 17 files changed, 510 insertions(+), 57 deletions(-) create mode 100644 packages/core/android/src/main/java/io/sentry/react/RNSentryTimeToDisplay.java create mode 100644 packages/core/ios/RNSentryTimeToDisplay.h create mode 100644 packages/core/ios/RNSentryTimeToDisplay.m create mode 100644 packages/core/src/js/utils/sentryeventemitterfallback.ts delete mode 100644 packages/core/test/utils/mockedSentryeventemitter.ts create mode 100644 packages/core/test/utils/mockedSentryeventemitterfallback.ts create mode 100644 packages/core/test/utils/sentryeventemitterfallback.test.ts diff --git a/CHANGELOG.md b/CHANGELOG.md index 138c9e519c..3118925cc1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,10 @@ ## Unreleased +### Fixes + +- Enhanced accuracy of time-to-display spans. ([#4189](https://github.com/getsentry/sentry-react-native/pull/4189)) + ### Dependencies - Bump JavaScript SDK from v8.34.0 to v8.35.0 ([#4196](https://github.com/getsentry/sentry-react-native/pull/4196)) diff --git a/packages/core/android/src/main/java/io/sentry/react/RNSentryModuleImpl.java b/packages/core/android/src/main/java/io/sentry/react/RNSentryModuleImpl.java index ee24c634c2..7472187dd4 100644 --- a/packages/core/android/src/main/java/io/sentry/react/RNSentryModuleImpl.java +++ b/packages/core/android/src/main/java/io/sentry/react/RNSentryModuleImpl.java @@ -126,10 +126,13 @@ public class RNSentryModuleImpl { /** Max trace file size in bytes. */ private long maxTraceFileSize = 5 * 1024 * 1024; + private final @NotNull SentryDateProvider dateProvider; + public RNSentryModuleImpl(ReactApplicationContext reactApplicationContext) { packageInfo = getPackageInfo(reactApplicationContext); this.reactApplicationContext = reactApplicationContext; this.emitNewFrameEvent = createEmitNewFrameEvent(); + this.dateProvider = new SentryAndroidDateProvider(); } private ReactApplicationContext getReactApplicationContext() { @@ -141,8 +144,6 @@ private ReactApplicationContext getReactApplicationContext() { } private @NotNull Runnable createEmitNewFrameEvent() { - final @NotNull SentryDateProvider dateProvider = new SentryAndroidDateProvider(); - return () -> { final SentryDate endDate = dateProvider.now(); WritableMap event = Arguments.createMap(); @@ -745,6 +746,10 @@ public void disableNativeFramesTracking() { } } + public void getNewScreenTimeToDisplay(Promise promise) { + RNSentryTimeToDisplay.getTimeToDisplay(promise, dateProvider); + } + private String getProfilingTracesDirPath() { if (cacheDirPath == null) { cacheDirPath = diff --git a/packages/core/android/src/main/java/io/sentry/react/RNSentryTimeToDisplay.java b/packages/core/android/src/main/java/io/sentry/react/RNSentryTimeToDisplay.java new file mode 100644 index 0000000000..b6fab45492 --- /dev/null +++ b/packages/core/android/src/main/java/io/sentry/react/RNSentryTimeToDisplay.java @@ -0,0 +1,42 @@ +package io.sentry.react; + +import android.os.Handler; +import android.os.Looper; +import android.view.Choreographer; +import com.facebook.react.bridge.Promise; +import io.sentry.SentryDate; +import io.sentry.SentryDateProvider; + +public final class RNSentryTimeToDisplay { + + private RNSentryTimeToDisplay() {} + + public static void getTimeToDisplay(Promise promise, SentryDateProvider dateProvider) { + Looper mainLooper = Looper.getMainLooper(); + + if (mainLooper == null) { + promise.reject( + "GetTimeToDisplay is not able to measure the time to display: Main looper not" + + " available."); + return; + } + + // Ensure the code runs on the main thread + new Handler(mainLooper) + .post( + () -> { + try { + Choreographer choreographer = Choreographer.getInstance(); + + // Invoke the callback after the frame is rendered + choreographer.postFrameCallback( + frameTimeNanos -> { + final SentryDate endDate = dateProvider.now(); + promise.resolve(endDate.nanoTimestamp() / 1e9); + }); + } catch (Exception exception) { + promise.reject("Failed to receive the instance of Choreographer", exception); + } + }); + } +} diff --git a/packages/core/android/src/newarch/java/io/sentry/react/RNSentryModule.java b/packages/core/android/src/newarch/java/io/sentry/react/RNSentryModule.java index 78f1911da4..6ea8542e8b 100644 --- a/packages/core/android/src/newarch/java/io/sentry/react/RNSentryModule.java +++ b/packages/core/android/src/newarch/java/io/sentry/react/RNSentryModule.java @@ -172,4 +172,9 @@ public String getCurrentReplayId() { public void crashedLastRun(Promise promise) { this.impl.crashedLastRun(promise); } + + @Override + public void getNewScreenTimeToDisplay(Promise promise) { + this.impl.getNewScreenTimeToDisplay(promise); + } } diff --git a/packages/core/android/src/oldarch/java/io/sentry/react/RNSentryModule.java b/packages/core/android/src/oldarch/java/io/sentry/react/RNSentryModule.java index c77d4218df..57fcbf0a73 100644 --- a/packages/core/android/src/oldarch/java/io/sentry/react/RNSentryModule.java +++ b/packages/core/android/src/oldarch/java/io/sentry/react/RNSentryModule.java @@ -172,4 +172,9 @@ public String getCurrentReplayId() { public void crashedLastRun(Promise promise) { this.impl.crashedLastRun(promise); } + + @ReactMethod() + public void getNewScreenTimeToDisplay(Promise promise) { + this.impl.getNewScreenTimeToDisplay(promise); + } } diff --git a/packages/core/ios/RNSentry.mm b/packages/core/ios/RNSentry.mm index a04733e7b5..3009e46c09 100644 --- a/packages/core/ios/RNSentry.mm +++ b/packages/core/ios/RNSentry.mm @@ -1,5 +1,6 @@ #import #import "RNSentry.h" +#import "RNSentryTimeToDisplay.h" #if __has_include() #import @@ -62,6 +63,7 @@ + (void)storeEnvelope:(SentryEnvelope *)envelope; @implementation RNSentry { bool sentHybridSdkDidBecomeActive; bool hasListeners; + RNSentryTimeToDisplay *_timeToDisplay; } - (dispatch_queue_t)methodQueue @@ -139,6 +141,8 @@ - (SentryOptions *_Nullable)createOptionsWithDictionary:(NSDictionary *_Nonnull) [mutableOptions removeObjectForKey:@"tracesSampler"]; [mutableOptions removeObjectForKey:@"enableTracing"]; + _timeToDisplay = [[RNSentryTimeToDisplay alloc] init]; + #if SENTRY_TARGET_REPLAY_SUPPORTED [RNSentryReplay updateOptions:mutableOptions]; #endif @@ -786,4 +790,9 @@ - (NSDictionary*) fetchNativeStackFramesBy: (NSArray*)instructionsAdd } #endif +RCT_EXPORT_METHOD(getNewScreenTimeToDisplay:(RCTPromiseResolveBlock)resolve + rejecter:(RCTPromiseRejectBlock)reject) { + [_timeToDisplay getTimeToDisplay:resolve]; +} + @end diff --git a/packages/core/ios/RNSentryTimeToDisplay.h b/packages/core/ios/RNSentryTimeToDisplay.h new file mode 100644 index 0000000000..fbb468cb23 --- /dev/null +++ b/packages/core/ios/RNSentryTimeToDisplay.h @@ -0,0 +1,7 @@ +#import + +@interface RNSentryTimeToDisplay : NSObject + +- (void)getTimeToDisplay:(RCTResponseSenderBlock)callback; + +@end diff --git a/packages/core/ios/RNSentryTimeToDisplay.m b/packages/core/ios/RNSentryTimeToDisplay.m new file mode 100644 index 0000000000..88e24eedc7 --- /dev/null +++ b/packages/core/ios/RNSentryTimeToDisplay.m @@ -0,0 +1,43 @@ +#import "RNSentryTimeToDisplay.h" +#import +#import + +@implementation RNSentryTimeToDisplay +{ + CADisplayLink *displayLink; + RCTResponseSenderBlock resolveBlock; +} + +// Rename requestAnimationFrame to getTimeToDisplay +- (void)getTimeToDisplay:(RCTResponseSenderBlock)callback +{ + // Store the resolve block to use in the callback. + resolveBlock = callback; + +#if TARGET_OS_IOS + // Create and add a display link to get the callback after the screen is rendered. + displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(handleDisplayLink:)]; + [displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes]; +#else + resolveBlock(@[]); // Return nothing if not iOS. +#endif +} + +#if TARGET_OS_IOS +- (void)handleDisplayLink:(CADisplayLink *)link { + // Get the current time + NSTimeInterval currentTime = [[NSDate date] timeIntervalSince1970] * 1000.0; // Convert to milliseconds + + // Ensure the callback is valid and pass the current time back + if (resolveBlock) { + resolveBlock(@[@(currentTime)]); // Call the callback with the current time + resolveBlock = nil; // Clear the block after it's called + } + + // Invalidate the display link to stop future callbacks + [displayLink invalidate]; + displayLink = nil; +} +#endif + +@end diff --git a/packages/core/src/js/NativeRNSentry.ts b/packages/core/src/js/NativeRNSentry.ts index a5a7652fba..c82debe900 100644 --- a/packages/core/src/js/NativeRNSentry.ts +++ b/packages/core/src/js/NativeRNSentry.ts @@ -9,6 +9,7 @@ import type { UnsafeObject } from './utils/rnlibrariesinterface'; export interface Spec extends TurboModule { addListener: (eventType: string) => void; removeListeners: (id: number) => void; + getNewScreenTimeToDisplay(): Promise; addBreadcrumb(breadcrumb: UnsafeObject): void; captureEnvelope( bytes: string, diff --git a/packages/core/src/js/tracing/reactnavigation.ts b/packages/core/src/js/tracing/reactnavigation.ts index 871485bd7b..ce2bb1032c 100644 --- a/packages/core/src/js/tracing/reactnavigation.ts +++ b/packages/core/src/js/tracing/reactnavigation.ts @@ -12,8 +12,9 @@ import { import type { Client, Integration, Span } from '@sentry/types'; import { isPlainObject, logger, timestampInSeconds } from '@sentry/utils'; -import type { NewFrameEvent, SentryEventEmitter } from '../utils/sentryeventemitter'; -import { createSentryEventEmitter, NewFrameEventName } from '../utils/sentryeventemitter'; +import type { NewFrameEvent } from '../utils/sentryeventemitter'; +import type { SentryEventEmitterFallback } from '../utils/sentryeventemitterfallback'; +import { createSentryFallbackEventEmitter } from '../utils/sentryeventemitterfallback'; import { isSentrySpan } from '../utils/span'; import { RN_GLOBAL_OBJ } from '../utils/worldwide'; import { NATIVE } from '../wrapper'; @@ -30,7 +31,6 @@ import { } from './span'; import { manualInitialDisplaySpans, startTimeToInitialDisplaySpan } from './timetodisplay'; import { setSpanDurationAsMeasurementOnSpan } from './utils'; - export const INTEGRATION_NAME = 'ReactNavigation'; const NAVIGATION_HISTORY_MAX_SIZE = 200; @@ -81,7 +81,7 @@ export const reactNavigationIntegration = ({ registerNavigationContainer: (navigationContainerRef: unknown) => void; } => { let navigationContainer: NavigationContainer | undefined; - let newScreenFrameEventEmitter: SentryEventEmitter | undefined; + let newScreenFrameEventEmitter: SentryEventEmitterFallback | undefined; let tracing: ReactNativeTracingIntegration | undefined; let idleSpanOptions: Parameters[1] = defaultIdleOptions; @@ -95,8 +95,8 @@ export const reactNavigationIntegration = ({ let recentRouteKeys: string[] = []; if (enableTimeToInitialDisplay) { - newScreenFrameEventEmitter = createSentryEventEmitter(); - newScreenFrameEventEmitter.initAsync(NewFrameEventName); + newScreenFrameEventEmitter = createSentryFallbackEventEmitter(); + newScreenFrameEventEmitter.initAsync(); NATIVE.initNativeReactNavigationNewFrameTracking().catch((reason: unknown) => { logger.error(`${INTEGRATION_NAME} Failed to initialize native new frame tracking: ${reason}`); }); @@ -258,9 +258,8 @@ export const reactNavigationIntegration = ({ }); const navigationSpanWithTtid = latestNavigationSpan; - !routeHasBeenSeen && - latestTtidSpan && - newScreenFrameEventEmitter?.once(NewFrameEventName, ({ newFrameTimestampInSeconds }: NewFrameEvent) => { + if (!routeHasBeenSeen && latestTtidSpan) { + newScreenFrameEventEmitter?.onceNewFrame(({ newFrameTimestampInSeconds }: NewFrameEvent) => { const activeSpan = getActiveSpan(); if (activeSpan && manualInitialDisplaySpans.has(activeSpan)) { logger.warn('[ReactNavigationInstrumentation] Detected manual instrumentation for the current active span.'); @@ -271,6 +270,7 @@ export const reactNavigationIntegration = ({ latestTtidSpan.end(newFrameTimestampInSeconds); setSpanDurationAsMeasurementOnSpan('time_to_initial_display', latestTtidSpan, navigationSpanWithTtid); }); + } navigationProcessingSpan?.updateName(`Processing navigation to ${route.name}`); navigationProcessingSpan?.setStatus({ code: SPAN_STATUS_OK }); diff --git a/packages/core/src/js/utils/sentryeventemitterfallback.ts b/packages/core/src/js/utils/sentryeventemitterfallback.ts new file mode 100644 index 0000000000..8eaeb947a8 --- /dev/null +++ b/packages/core/src/js/utils/sentryeventemitterfallback.ts @@ -0,0 +1,98 @@ +import { logger, timestampInSeconds } from '@sentry/utils'; + +import { NATIVE } from '../wrapper'; +import type { NewFrameEvent, SentryEventEmitter } from './sentryeventemitter'; +import { createSentryEventEmitter, NewFrameEventName } from './sentryeventemitter'; + +export const FALLBACK_TIMEOUT_MS = 10_000; + +export type FallBackNewFrameEvent = { newFrameTimestampInSeconds: number; isFallback?: boolean }; +export interface SentryEventEmitterFallback { + /** + * Initializes the fallback event emitter + * This method is synchronous in JS but the event emitter starts asynchronously. + */ + initAsync: () => void; + onceNewFrame: (listener: (event: FallBackNewFrameEvent) => void) => void; +} + +/** + * Creates emitter that allows to listen to UI Frame events when ready. + */ +export function createSentryFallbackEventEmitter( + emitter: SentryEventEmitter = createSentryEventEmitter(), + fallbackTimeoutMs = FALLBACK_TIMEOUT_MS, +): SentryEventEmitterFallback { + let fallbackTimeout: ReturnType | undefined; + let animationFrameTimestampSeconds: number | undefined; + let nativeNewFrameTimestampSeconds: number | undefined; + + function getAnimationFrameTimestampSeconds(): void { + // https://reactnative.dev/docs/timers#timers + // NOTE: The current implementation of requestAnimationFrame is the same + // as setTimeout(0). This isn't exactly how requestAnimationFrame + // is supposed to work on web, so it doesn't get called when UI Frames are rendered.: https://github.com/facebook/react-native/blob/5106933c750fee2ce49fe1945c3e3763eebc92bc/packages/react-native/ReactCommon/react/runtime/TimerManager.cpp#L442-L443 + requestAnimationFrame(() => { + if (fallbackTimeout === undefined) { + return; + } + animationFrameTimestampSeconds = timestampInSeconds(); + }); + } + + function getNativeNewFrameTimestampSeconds(): void { + NATIVE.getNewScreenTimeToDisplay() + .then(resolve => { + if (fallbackTimeout === undefined) { + return; + } + nativeNewFrameTimestampSeconds = resolve ?? undefined; + }) + .catch(reason => { + logger.error('Failed to receive Native fallback timestamp.', reason); + }); + } + + return { + initAsync() { + emitter.initAsync(NewFrameEventName); + }, + + onceNewFrame(listener: (event: FallBackNewFrameEvent) => void) { + animationFrameTimestampSeconds = undefined; + nativeNewFrameTimestampSeconds = undefined; + + const internalListener = (event: NewFrameEvent): void => { + if (fallbackTimeout !== undefined) { + clearTimeout(fallbackTimeout); + fallbackTimeout = undefined; + } + animationFrameTimestampSeconds = undefined; + nativeNewFrameTimestampSeconds = undefined; + listener(event); + }; + fallbackTimeout = setTimeout(() => { + if (nativeNewFrameTimestampSeconds) { + logger.log('Native event emitter did not reply in time'); + return listener({ + newFrameTimestampInSeconds: nativeNewFrameTimestampSeconds, + isFallback: true, + }); + } else if (animationFrameTimestampSeconds) { + logger.log('[Sentry] Native event emitter did not reply in time. Using JavaScript fallback emitter.'); + return listener({ + newFrameTimestampInSeconds: animationFrameTimestampSeconds, + isFallback: true, + }); + } else { + emitter.removeListener(NewFrameEventName, internalListener); + logger.error('Failed to receive any fallback timestamp.'); + } + }, fallbackTimeoutMs); + + getNativeNewFrameTimestampSeconds(); + getAnimationFrameTimestampSeconds(); + emitter.once(NewFrameEventName, internalListener); + }, + }; +} diff --git a/packages/core/src/js/wrapper.ts b/packages/core/src/js/wrapper.ts index d901524793..577eaf99e6 100644 --- a/packages/core/src/js/wrapper.ts +++ b/packages/core/src/js/wrapper.ts @@ -116,6 +116,7 @@ interface SentryNativeWrapper { getCurrentReplayId(): string | null; crashedLastRun(): Promise; + getNewScreenTimeToDisplay(): Promise; } const EOL = utf8ToBytes('\n'); @@ -690,6 +691,14 @@ export const NATIVE: SentryNativeWrapper = { return typeof result === 'boolean' ? result : null; }, + getNewScreenTimeToDisplay(): Promise { + if (!this.enableNative || !this._isModuleLoaded(RNSentry)) { + return Promise.resolve(null); + } + + return RNSentry.getNewScreenTimeToDisplay(); + }, + /** * Gets the event from envelopeItem and applies the level filter to the selected event. * @param data An envelope item containing the event. diff --git a/packages/core/test/mockWrapper.ts b/packages/core/test/mockWrapper.ts index 58387bb8da..82b2b9194c 100644 --- a/packages/core/test/mockWrapper.ts +++ b/packages/core/test/mockWrapper.ts @@ -58,6 +58,7 @@ const NATIVE: MockInterface = { getCurrentReplayId: jest.fn(), crashedLastRun: jest.fn(), + getNewScreenTimeToDisplay: jest.fn().mockResolvedValue(42), }; NATIVE.isNativeAvailable.mockReturnValue(true); diff --git a/packages/core/test/tracing/reactnavigation.ttid.test.tsx b/packages/core/test/tracing/reactnavigation.ttid.test.tsx index 78dacd003c..beb8a1139a 100644 --- a/packages/core/test/tracing/reactnavigation.ttid.test.tsx +++ b/packages/core/test/tracing/reactnavigation.ttid.test.tsx @@ -4,11 +4,11 @@ import * as TestRenderer from '@testing-library/react-native' import * as React from "react"; import * as mockWrapper from '../mockWrapper'; -import * as mockedSentryEventEmitter from '../utils/mockedSentryeventemitter'; +import * as mockedSentryEventEmitter from '../utils/mockedSentryeventemitterfallback'; import * as mockedtimetodisplaynative from './mockedtimetodisplaynative'; jest.mock('../../src/js/wrapper', () => mockWrapper); jest.mock('../../src/js/utils/environment'); -jest.mock('../../src/js/utils/sentryeventemitter', () => mockedSentryEventEmitter); +jest.mock('../../src/js/utils/sentryeventemitterfallback', () => mockedSentryEventEmitter); jest.mock('../../src/js/tracing/timetodisplaynative', () => mockedtimetodisplaynative); import * as Sentry from '../../src/js'; @@ -17,11 +17,11 @@ import { TimeToFullDisplay, TimeToInitialDisplay } from '../../src/js/tracing'; import { _setAppStartEndTimestampMs } from '../../src/js/tracing/integrations/appStart'; import { SPAN_ORIGIN_AUTO_NAVIGATION_REACT_NAVIGATION, SPAN_ORIGIN_AUTO_UI_TIME_TO_DISPLAY, SPAN_ORIGIN_MANUAL_UI_TIME_TO_DISPLAY } from '../../src/js/tracing/origin'; import { isHermesEnabled, notWeb } from '../../src/js/utils/environment'; -import { createSentryEventEmitter } from '../../src/js/utils/sentryeventemitter'; +import { createSentryFallbackEventEmitter } from '../../src/js/utils/sentryeventemitterfallback'; import { RN_GLOBAL_OBJ } from '../../src/js/utils/worldwide'; import { MOCK_DSN } from '../mockDsn'; import { secondInFutureTimestampMs } from '../testutils'; -import type { MockedSentryEventEmitter } from '../utils/mockedSentryeventemitter'; +import type { MockedSentryEventEmitterFallback } from '../utils/mockedSentryeventemitterfallback'; import { emitNativeFullDisplayEvent, emitNativeInitialDisplayEvent } from './mockedtimetodisplaynative'; import { createMockNavigationAndAttachTo } from './reactnavigationutils'; @@ -32,7 +32,7 @@ type ScopeWithMaybeSpan = Scope & { }; describe('React Navigation - TTID', () => { - let mockedEventEmitter: MockedSentryEventEmitter; + let mockedEventEmitter: MockedSentryEventEmitterFallback; let transportSendMock: jest.Mock, Parameters>; let mockedNavigation: ReturnType; const mockedAppStartTimeSeconds: number = timestampInSeconds(); @@ -51,8 +51,8 @@ describe('React Navigation - TTID', () => { }); _setAppStartEndTimestampMs(mockedAppStartTimeSeconds * 1000); - mockedEventEmitter = mockedSentryEventEmitter.createMockedSentryEventEmitter(); - (createSentryEventEmitter as jest.Mock).mockReturnValue(mockedEventEmitter); + mockedEventEmitter = mockedSentryEventEmitter.createMockedSentryFallbackEventEmitter(); + (createSentryFallbackEventEmitter as jest.Mock).mockReturnValue(mockedEventEmitter); const sut = createTestedInstrumentation({ enableTimeToInitialDisplay: true }); transportSendMock = initSentry(sut).transportSendMock; diff --git a/packages/core/test/utils/mockedSentryeventemitter.ts b/packages/core/test/utils/mockedSentryeventemitter.ts deleted file mode 100644 index 6849e460be..0000000000 --- a/packages/core/test/utils/mockedSentryeventemitter.ts +++ /dev/null @@ -1,39 +0,0 @@ -import { timestampInSeconds } from '@sentry/utils'; -import * as EventEmitter from 'events'; - -import type { NewFrameEvent, SentryEventEmitter } from '../../src/js/utils/sentryeventemitter'; -import type { MockInterface } from '../testutils'; - -export const NewFrameEventName = 'rn_sentry_new_frame'; -export type NewFrameEventName = typeof NewFrameEventName; - -export interface MockedSentryEventEmitter extends MockInterface { - emitNewFrameEvent: (timestampSeconds?: number) => void; -} - -export function createMockedSentryEventEmitter(): MockedSentryEventEmitter { - const emitter = new EventEmitter(); - - return { - emitNewFrameEvent: jest.fn((timestampSeconds?: number) => { - emitter.emit('rn_sentry_new_frame', { - newFrameTimestampInSeconds: timestampSeconds || timestampInSeconds(), - }); - }), - once: jest.fn((event: NewFrameEventName, listener: (event: NewFrameEvent) => void) => { - emitter.once(event, listener); - }), - removeListener: jest.fn((event: NewFrameEventName, listener: (event: NewFrameEvent) => void) => { - emitter.removeListener(event, listener); - }), - addListener: jest.fn((event: NewFrameEventName, listener: (event: NewFrameEvent) => void) => { - emitter.addListener(event, listener); - }), - initAsync: jest.fn(), - closeAllAsync: jest.fn(() => { - emitter.removeAllListeners(); - }), - }; -} - -export const createSentryEventEmitter = jest.fn(() => createMockedSentryEventEmitter()); diff --git a/packages/core/test/utils/mockedSentryeventemitterfallback.ts b/packages/core/test/utils/mockedSentryeventemitterfallback.ts new file mode 100644 index 0000000000..481b109d68 --- /dev/null +++ b/packages/core/test/utils/mockedSentryeventemitterfallback.ts @@ -0,0 +1,27 @@ +import { timestampInSeconds } from '@sentry/utils'; +import * as EventEmitter from 'events'; + +import type { NewFrameEvent } from '../../src/js/utils/sentryeventemitter'; +import type { SentryEventEmitterFallback } from '../../src/js/utils/sentryeventemitterfallback'; +import type { MockInterface } from '../testutils'; + +export const NewFrameEventName = 'rn_sentry_new_frame'; +export type NewFrameEventName = typeof NewFrameEventName; +export interface MockedSentryEventEmitterFallback extends MockInterface { + emitNewFrameEvent: (timestampSeconds?: number) => void; +} +export function createMockedSentryFallbackEventEmitter(): MockedSentryEventEmitterFallback { + const emitter = new EventEmitter(); + return { + initAsync: jest.fn(), + emitNewFrameEvent: jest.fn((timestampSeconds?: number) => { + emitter.emit(NewFrameEventName, { + newFrameTimestampInSeconds: timestampSeconds || timestampInSeconds(), + }); + }), + onceNewFrame: jest.fn((listener: (event: NewFrameEvent) => void) => { + emitter.once(NewFrameEventName, listener); + }), + }; +} +export const createSentryFallbackEventEmitter = jest.fn(() => createMockedSentryFallbackEventEmitter()); diff --git a/packages/core/test/utils/sentryeventemitterfallback.test.ts b/packages/core/test/utils/sentryeventemitterfallback.test.ts new file mode 100644 index 0000000000..e96567ae29 --- /dev/null +++ b/packages/core/test/utils/sentryeventemitterfallback.test.ts @@ -0,0 +1,236 @@ +import { logger } from '@sentry/utils'; + +import { NewFrameEventName } from '../../src/js/utils/sentryeventemitter'; +import { createSentryFallbackEventEmitter } from '../../src/js/utils/sentryeventemitterfallback'; + +// Mock dependencies +jest.mock('../../src/js/utils/environment', () => ({ + isTurboModuleEnabled: () => false, +})); + +jest.mock('../../src/js/wrapper', () => jest.requireActual('../mockWrapper')); + +import { NATIVE } from '../../src/js/wrapper'; + +jest.spyOn(logger, 'warn'); +jest.spyOn(logger, 'log'); +jest.spyOn(logger, 'error'); + +describe('SentryEventEmitterFallback', () => { + let emitter: ReturnType; + + beforeEach(() => { + jest.clearAllMocks(); + // @ts-expect-error test + jest.spyOn(window, 'requestAnimationFrame').mockImplementation(cb => cb()); + emitter = createSentryFallbackEventEmitter(); + NATIVE.getNewScreenTimeToDisplay = jest.fn(() => Promise.resolve(12345)); + }); + + afterEach(() => { + // @ts-expect-error test + window.requestAnimationFrame.mockRestore(); + NATIVE.getNewScreenTimeToDisplay = jest.fn(); + }); + + it('should start listener and use fallback when native call returned undefined/null', async () => { + jest.useFakeTimers(); + const spy = jest.spyOn(require('@sentry/utils'), 'timestampInSeconds'); + const fallbackTime = Date.now() / 1000; + spy.mockReturnValue(fallbackTime); + + (NATIVE.getNewScreenTimeToDisplay as jest.Mock).mockReturnValue(Promise.resolve()); + + const listener = jest.fn(); + emitter.onceNewFrame(listener); + + // Wait for the next event loop to allow startListenerAsync to call NATIVE.getNewScreenTimeToDisplay + await Promise.resolve(); + + await expect(NATIVE.getNewScreenTimeToDisplay).toHaveBeenCalled(); + expect(logger.error).not.toHaveBeenCalledWith('Failed to receive Native fallback timestamp.', expect.any(Error)); + + // Simulate retries and timer + jest.runAllTimers(); + + // Ensure fallback event is emitted + expect(listener).toHaveBeenCalledWith({ + newFrameTimestampInSeconds: fallbackTime, + isFallback: true, + }); + expect(logger.log).toHaveBeenCalledWith( + expect.stringContaining( + '[Sentry] Native event emitter did not reply in time. Using JavaScript fallback emitter.', + ), + ); + }); + + it('should start listener and use fallback when native call fails', async () => { + jest.useFakeTimers(); + + (NATIVE.getNewScreenTimeToDisplay as jest.Mock).mockRejectedValue(new Error('Failed')); + + const spy = jest.spyOn(require('@sentry/utils'), 'timestampInSeconds'); + const fallbackTime = Date.now() / 1000; + spy.mockReturnValue(fallbackTime); + + const listener = jest.fn(); + emitter.onceNewFrame(listener); + + // Wait for the next event loop to allow startListenerAsync to call NATIVE.getNewScreenTimeToDisplay + await Promise.resolve(); + + await expect(NATIVE.getNewScreenTimeToDisplay).toHaveBeenCalled(); + expect(logger.error).toHaveBeenCalledWith('Failed to receive Native fallback timestamp.', expect.any(Error)); + + // Simulate retries and timer + jest.runAllTimers(); + + // Ensure fallback event is emitted + expect(listener).toHaveBeenCalledWith({ + newFrameTimestampInSeconds: fallbackTime, + isFallback: true, + }); + expect(logger.log).toHaveBeenCalledWith( + expect.stringContaining( + '[Sentry] Native event emitter did not reply in time. Using JavaScript fallback emitter.', + ), + ); + }); + + it('should start listener and use fallback when native call fails', async () => { + jest.useFakeTimers(); + const spy = jest.spyOn(require('@sentry/utils'), 'timestampInSeconds'); + const fallbackTime = Date.now() / 1000; + spy.mockReturnValue(fallbackTime); + + (NATIVE.getNewScreenTimeToDisplay as jest.Mock).mockRejectedValue(new Error('Failed')); + + const listener = jest.fn(); + emitter.onceNewFrame(listener); + + // Wait for the next event loop to allow startListenerAsync to call NATIVE.getNewScreenTimeToDisplay + await Promise.resolve(); + + await expect(NATIVE.getNewScreenTimeToDisplay).toHaveBeenCalled(); + expect(logger.error).toHaveBeenCalledWith('Failed to receive Native fallback timestamp.', expect.any(Error)); + + // Simulate retries and timer + jest.runAllTimers(); + + // Ensure fallback event is emitted + expect(listener).toHaveBeenCalledWith({ + newFrameTimestampInSeconds: fallbackTime, + isFallback: true, + }); + expect(logger.log).toHaveBeenCalledWith( + expect.stringContaining( + '[Sentry] Native event emitter did not reply in time. Using JavaScript fallback emitter.', + ), + ); + }); + + it('should start listener and use fallback when native call is not available', async () => { + jest.useFakeTimers(); + const spy = jest.spyOn(require('@sentry/utils'), 'timestampInSeconds'); + const fallbackTime = Date.now() / 1000; + spy.mockReturnValue(fallbackTime); + + NATIVE.getNewScreenTimeToDisplay = () => Promise.resolve(null); + + const listener = jest.fn(); + emitter.onceNewFrame(listener); + + // Wait for the next event loop to allow startListenerAsync to call NATIVE.getNewScreenTimeToDisplay + await Promise.resolve(); + + // Simulate retries and timer + jest.runAllTimers(); + + // Ensure fallback event is emitted + expect(listener).toHaveBeenCalledWith({ + newFrameTimestampInSeconds: fallbackTime, + isFallback: true, + }); + expect(logger.log).toHaveBeenCalledWith( + expect.stringContaining( + '[Sentry] Native event emitter did not reply in time. Using JavaScript fallback emitter.', + ), + ); + }); + + it('should start listener and call native when native module is available', async () => { + const nativeTimestamp = 12345; + + (NATIVE.getNewScreenTimeToDisplay as jest.Mock).mockResolvedValueOnce(nativeTimestamp); + + const listener = jest.fn(); + emitter.onceNewFrame(listener); + + expect(NATIVE.getNewScreenTimeToDisplay).toHaveBeenCalled(); + }); + + it('should not emit if original event emitter was called', async () => { + jest.useFakeTimers(); + + // Capture the callback passed to addListener + // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/ban-types + let callback: Function = () => {}; + const mockOnce = jest.fn().mockImplementationOnce((eventName, cb) => { + if (eventName === NewFrameEventName) { + callback = cb; + } + return { + remove: jest.fn(), + }; + }); + + emitter = createSentryFallbackEventEmitter({ + addListener: jest.fn(), + initAsync: jest.fn(), + closeAllAsync: jest.fn(), + removeListener: jest.fn(), + once: mockOnce, + }); + + emitter.initAsync(); + const listener = jest.fn(); + emitter.onceNewFrame(listener); + callback({ + newFrameTimestampInSeconds: 67890, + }); + + // Wait for the next event loop to allow startListenerAsync to call NATIVE.getNewScreenTimeToDisplay + await Promise.resolve(); + + // Simulate retries and timer + jest.runAllTimers(); + + // Ensure fallback event is emitted + expect(listener).toHaveBeenCalledWith({ + newFrameTimestampInSeconds: 67890, + isFallback: undefined, + }); + expect(logger.log).not.toBeCalled(); + }); + + it('should retry up to maxRetries and emit fallback if no response', async () => { + jest.useFakeTimers(); + + const listener = jest.fn(); + emitter.onceNewFrame(listener); + + // Wait for the next event loop to allow startListenerAsync to call NATIVE.getNewScreenTimeToDisplay + await Promise.resolve(); + + expect(logger.log).not.toHaveBeenCalled(); + + // Simulate retries and timer + jest.runAllTimers(); + + expect(listener).toHaveBeenCalledWith(expect.objectContaining({ isFallback: true })); + expect(logger.log).toHaveBeenCalledWith(expect.stringContaining('Native event emitter did not reply in time')); + + jest.useRealTimers(); + }); +});