From fa2b6e93b6efa33a316bd3aaa0379662ec8aee82 Mon Sep 17 00:00:00 2001 From: Krystof Woldrich Date: Thu, 25 Jan 2024 00:02:44 -0500 Subject: [PATCH 1/6] feat(integrations): Add Spotlight integration with auto dev server hostname detection --- samples/expo/app/(tabs)/index.tsx | 1 + samples/react-native/src/App.tsx | 1 + src/js/integrations/index.ts | 1 + src/js/integrations/spotlight.ts | 98 +++++++++++++++++++++++++++++++ 4 files changed, 101 insertions(+) create mode 100644 src/js/integrations/spotlight.ts diff --git a/samples/expo/app/(tabs)/index.tsx b/samples/expo/app/(tabs)/index.tsx index d5b961222b..83add11213 100644 --- a/samples/expo/app/(tabs)/index.tsx +++ b/samples/expo/app/(tabs)/index.tsx @@ -38,6 +38,7 @@ Sentry.init({ // default: [/.*/] failedRequestTargets: [/.*/], }), + Sentry.Integrations.Spotlight(), ); return integrations.filter(i => i.name !== 'Dedupe'); }, diff --git a/samples/react-native/src/App.tsx b/samples/react-native/src/App.tsx index 385692d11b..d4f57a71cf 100644 --- a/samples/react-native/src/App.tsx +++ b/samples/react-native/src/App.tsx @@ -69,6 +69,7 @@ Sentry.init({ // default: [/.*/] failedRequestTargets: [/.*/], }), + Sentry.Integrations.Spotlight(), ); return integrations.filter(i => i.name !== 'Dedupe'); }, diff --git a/src/js/integrations/index.ts b/src/js/integrations/index.ts index 061bc5e0a9..3a8ad303ae 100644 --- a/src/js/integrations/index.ts +++ b/src/js/integrations/index.ts @@ -7,3 +7,4 @@ export { SdkInfo } from './sdkinfo'; export { ReactNativeInfo } from './reactnativeinfo'; export { ModulesLoader } from './modulesloader'; export { HermesProfiling } from '../profiling/integration'; +export { Spotlight } from './spotlight'; diff --git a/src/js/integrations/spotlight.ts b/src/js/integrations/spotlight.ts new file mode 100644 index 0000000000..c6238ae3db --- /dev/null +++ b/src/js/integrations/spotlight.ts @@ -0,0 +1,98 @@ +import type { Client, Envelope, EventProcessor, Integration } from '@sentry/types'; +import { logger, serializeEnvelope } from '@sentry/utils'; + +import { makeUtf8TextEncoder } from '../transports/TextEncoder'; +import { ReactNativeLibraries } from '../utils/rnlibraries'; + +type SpotlightReactNativeIntegrationOptions = { + /** + * The URL of the Sidecar instance to connect and forward events to. + * If not set, Spotlight will try to connect to the Sidecar running on localhost:8969. + * + * @default "http://localhost:8969/stream" + */ + sidecarUrl?: string; +}; + +/** + * + */ +export function Spotlight({ + sidecarUrl = getDefaultSidecarUrl(), +}: SpotlightReactNativeIntegrationOptions = {}): Integration { + logger.info('[Spotlight] Using Sidecar URL', sidecarUrl); + + return { + name: 'Spotlight', + + setupOnce(_: (callback: EventProcessor) => void, getCurrentHub) { + const client = getCurrentHub().getClient(); + if (client) { + setup(client, sidecarUrl); + } else { + logger.warn('[Spotlight] Could not initialize Sidecar integration due to missing Client'); + } + }, + }; +}; + +function setup(client: Client, sidecarUrl: string): void { + sendEnvelopesToSidecar(client, sidecarUrl); +}; + +function sendEnvelopesToSidecar(client: Client, sidecarUrl: string): void { + // Ensure, integrations are initialized even if no DSN was set + // client.setupIntegrations(true); + + if (!client.on) { + return; + } + + client.on('beforeEnvelope', (envelope: Envelope) => { + // TODO: This is a workaround for spotlight/sidecar not supporting images + const envelopeItems= [...envelope[1]].filter(item => + typeof item[0].content_type !== 'string' || + !item[0].content_type.startsWith('image')); + + envelope[1] = envelopeItems as Envelope[1]; + + fetch(sidecarUrl, { + method: 'POST', + body: serializeEnvelope(envelope, makeUtf8TextEncoder()), + headers: { + 'Content-Type': 'application/x-sentry-envelope', + }, + mode: 'cors', + }).catch(err => { + logger.error( + '[Spotlight] Sentry SDK can\'t connect to Sidecar is it running? See: https://spotlightjs.com/sidecar/npx/', + err, + ); + }); + }); +} + +function getDefaultSidecarUrl(): string { + try { + const { url } = ReactNativeLibraries.Devtools?.getDevServer(); + return `http://${getHostnameFromString(url)}:8969/stream`; + } catch (_oO) { + // We can't load devserver URL + } + return 'http://localhost:8969/stream'; +} + +/** + * React Native implementation of the URL class is missing the `hostname` property. + */ +function getHostnameFromString(urlString: string): string | null { + const regex = /^(?:\w+:)?\/\/([^/:]+)(:\d+)?(.*)$/; + const matches = urlString.match(regex); + + if (matches && matches[1]) { + return matches[1]; + } else { + // Invalid URL format + return null; + } +} From cbe81719e8590b003effdf5c7e57a2da28d34ab0 Mon Sep 17 00:00:00 2001 From: Krystof Woldrich Date: Thu, 25 Jan 2024 09:01:56 -0500 Subject: [PATCH 2/6] fix envelope overwrite --- src/js/integrations/spotlight.ts | 22 ++++++++++------------ 1 file changed, 10 insertions(+), 12 deletions(-) diff --git a/src/js/integrations/spotlight.ts b/src/js/integrations/spotlight.ts index c6238ae3db..d07c97e710 100644 --- a/src/js/integrations/spotlight.ts +++ b/src/js/integrations/spotlight.ts @@ -34,38 +34,36 @@ export function Spotlight({ } }, }; -}; +} function setup(client: Client, sidecarUrl: string): void { sendEnvelopesToSidecar(client, sidecarUrl); -}; +} function sendEnvelopesToSidecar(client: Client, sidecarUrl: string): void { - // Ensure, integrations are initialized even if no DSN was set - // client.setupIntegrations(true); - if (!client.on) { return; } - client.on('beforeEnvelope', (envelope: Envelope) => { + client.on('beforeEnvelope', (originalEnvelope: Envelope) => { // TODO: This is a workaround for spotlight/sidecar not supporting images - const envelopeItems= [...envelope[1]].filter(item => - typeof item[0].content_type !== 'string' || - !item[0].content_type.startsWith('image')); + const spotlightEnvelope: Envelope = [...originalEnvelope]; + const envelopeItems = [...originalEnvelope[1]].filter( + item => typeof item[0].content_type !== 'string' || !item[0].content_type.startsWith('image'), + ); - envelope[1] = envelopeItems as Envelope[1]; + spotlightEnvelope[1] = envelopeItems as Envelope[1]; fetch(sidecarUrl, { method: 'POST', - body: serializeEnvelope(envelope, makeUtf8TextEncoder()), + body: serializeEnvelope(spotlightEnvelope, makeUtf8TextEncoder()), headers: { 'Content-Type': 'application/x-sentry-envelope', }, mode: 'cors', }).catch(err => { logger.error( - '[Spotlight] Sentry SDK can\'t connect to Sidecar is it running? See: https://spotlightjs.com/sidecar/npx/', + "[Spotlight] Sentry SDK can't connect to Sidecar is it running? See: https://spotlightjs.com/sidecar/npx/", err, ); }); From 8a02b593ad865ced902291ce05c477d2dac9bbb8 Mon Sep 17 00:00:00 2001 From: Krystof Woldrich Date: Thu, 25 Jan 2024 11:05:38 -0500 Subject: [PATCH 3/6] add new connection error message --- src/js/integrations/spotlight.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/js/integrations/spotlight.ts b/src/js/integrations/spotlight.ts index d07c97e710..e533c3cabf 100644 --- a/src/js/integrations/spotlight.ts +++ b/src/js/integrations/spotlight.ts @@ -63,7 +63,7 @@ function sendEnvelopesToSidecar(client: Client, sidecarUrl: string): void { mode: 'cors', }).catch(err => { logger.error( - "[Spotlight] Sentry SDK can't connect to Sidecar is it running? See: https://spotlightjs.com/sidecar/npx/", + "[Spotlight] Sentry SDK can't connect to Spotlight is it running? See https://spotlightjs.com to download it.", err, ); }); From da8c265ccb11501a6e7c99f7c86caf38ab77d5da Mon Sep 17 00:00:00 2001 From: Krystof Woldrich Date: Fri, 26 Jan 2024 17:43:58 -0500 Subject: [PATCH 4/6] add spotlight changelog --- CHANGELOG.md | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 259af1f881..9e143eed2f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,21 @@ ## Unreleased +### Features + +- Add [`@spotlightjs/spotlight`](https://spotlightjs.com/) support ([#3559](https://github.com/getsentry/sentry-react-native/pull/3559)) + + Download the `Spotlight` desktop application and add the integration to your `Sentry.init`. + + ```javascript + import * as Sentry from '@sentry/react-native'; + + Sentry.init({ + dsn: '___DSN___', + integrations: __DEV__ ? [Sentry.Integrations.Spotlight()] : [], + }); + ``` + ### Fixes - Prevent pod install crash when visionos is not present ([#3548](https://github.com/getsentry/sentry-react-native/pull/3548)) From 955b3997b43e281f0e3c4ca93a4907c66be4675d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kry=C5=A1tof=20Wold=C5=99ich?= <31292499+krystofwoldrich@users.noreply.github.com> Date: Fri, 26 Jan 2024 17:51:58 -0500 Subject: [PATCH 5/6] Update CHANGELOG.md --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 9e143eed2f..5504b173a0 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,7 +4,7 @@ ### Features -- Add [`@spotlightjs/spotlight`](https://spotlightjs.com/) support ([#3559](https://github.com/getsentry/sentry-react-native/pull/3559)) +- Add [`@spotlightjs/spotlight`](https://spotlightjs.com/) support ([#3550](https://github.com/getsentry/sentry-react-native/pull/3550)) Download the `Spotlight` desktop application and add the integration to your `Sentry.init`. From 4424579cd0bdef9c1c232713f49d196598c101f4 Mon Sep 17 00:00:00 2001 From: Krystof Woldrich Date: Fri, 26 Jan 2024 19:28:12 -0500 Subject: [PATCH 6/6] add tests and easy to setup options --- CHANGELOG.md | 2 +- jest.config.js | 2 +- package.json | 1 + samples/expo/app/(tabs)/index.tsx | 2 +- samples/react-native/src/App.tsx | 2 +- src/js/integrations/default.ts | 9 +++ src/js/integrations/spotlight.ts | 2 + src/js/options.ts | 21 +++++++ test/integrations/spotlight.test.ts | 97 +++++++++++++++++++++++++++++ test/mockFetch.ts | 2 + test/sdk.test.ts | 18 ++++++ yarn.lock | 15 ++++- 12 files changed, 168 insertions(+), 5 deletions(-) create mode 100644 test/integrations/spotlight.test.ts create mode 100644 test/mockFetch.ts diff --git a/CHANGELOG.md b/CHANGELOG.md index 9e143eed2f..a829e29fc8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -13,7 +13,7 @@ Sentry.init({ dsn: '___DSN___', - integrations: __DEV__ ? [Sentry.Integrations.Spotlight()] : [], + enableSpotlight: __DEV__, }); ``` diff --git a/jest.config.js b/jest.config.js index d8df7d57a8..e559f52c25 100644 --- a/jest.config.js +++ b/jest.config.js @@ -1,7 +1,7 @@ module.exports = { collectCoverage: true, preset: 'react-native', - setupFilesAfterEnv: ['/test/mockConsole.ts'], + setupFilesAfterEnv: ['/test/mockConsole.ts', '/test/mockFetch.ts'], globals: { __DEV__: true, 'ts-jest': { diff --git a/package.json b/package.json index 186a2dd3ad..6a5ca98243 100644 --- a/package.json +++ b/package.json @@ -96,6 +96,7 @@ "expo-module-scripts": "^3.1.0", "jest": "^29.6.2", "jest-environment-jsdom": "^29.6.2", + "jest-fetch-mock": "^3.0.3", "madge": "^6.1.0", "metro": "0.76", "prettier": "^2.0.5", diff --git a/samples/expo/app/(tabs)/index.tsx b/samples/expo/app/(tabs)/index.tsx index 83add11213..9a5b51c0b4 100644 --- a/samples/expo/app/(tabs)/index.tsx +++ b/samples/expo/app/(tabs)/index.tsx @@ -38,7 +38,6 @@ Sentry.init({ // default: [/.*/] failedRequestTargets: [/.*/], }), - Sentry.Integrations.Spotlight(), ); return integrations.filter(i => i.name !== 'Dedupe'); }, @@ -63,6 +62,7 @@ Sentry.init({ _experiments: { profilesSampleRate: 0, }, + enableSpotlight: true, }); export default function TabOneScreen() { diff --git a/samples/react-native/src/App.tsx b/samples/react-native/src/App.tsx index d4f57a71cf..bfc0573ef7 100644 --- a/samples/react-native/src/App.tsx +++ b/samples/react-native/src/App.tsx @@ -69,7 +69,6 @@ Sentry.init({ // default: [/.*/] failedRequestTargets: [/.*/], }), - Sentry.Integrations.Spotlight(), ); return integrations.filter(i => i.name !== 'Dedupe'); }, @@ -94,6 +93,7 @@ Sentry.init({ _experiments: { profilesSampleRate: 0, }, + enableSpotlight: true, }); const Stack = createStackNavigator(); diff --git a/src/js/integrations/default.ts b/src/js/integrations/default.ts index 7b50b564da..4dc16bfae1 100644 --- a/src/js/integrations/default.ts +++ b/src/js/integrations/default.ts @@ -18,6 +18,7 @@ import { Release } from './release'; import { createReactNativeRewriteFrames } from './rewriteframes'; import { Screenshot } from './screenshot'; import { SdkInfo } from './sdkinfo'; +import { Spotlight } from './spotlight'; import { ViewHierarchy } from './viewhierarchy'; /** @@ -94,5 +95,13 @@ export function getDefaultIntegrations(options: ReactNativeClientOptions): Integ integrations.push(new ExpoContext()); } + if (options.enableSpotlight) { + integrations.push( + Spotlight({ + sidecarUrl: options.spotlightSidecarUrl, + }), + ); + } + return integrations; } diff --git a/src/js/integrations/spotlight.ts b/src/js/integrations/spotlight.ts index e533c3cabf..149cdb56ea 100644 --- a/src/js/integrations/spotlight.ts +++ b/src/js/integrations/spotlight.ts @@ -15,7 +15,9 @@ type SpotlightReactNativeIntegrationOptions = { }; /** + * Use this integration to send errors and transactions to Spotlight. * + * Learn more about spotlight at https://spotlightjs.com */ export function Spotlight({ sidecarUrl = getDefaultSidecarUrl(), diff --git a/src/js/options.ts b/src/js/options.ts index f2644953e3..d12db7991f 100644 --- a/src/js/options.ts +++ b/src/js/options.ts @@ -159,6 +159,27 @@ export interface BaseReactNativeOptions { * @default false */ enableCaptureFailedRequests?: boolean; + + /** + * This option will enable forwarding captured Sentry events to Spotlight. + * + * More details: https://spotlightjs.com/ + * + * IMPORTANT: Only set this option to `true` while developing, not in production! + */ + enableSpotlight?: boolean; + + /** + * This option changes the default Spotlight Sidecar URL. + * + * By default, the SDK expects the Sidecar to be running + * on the same host as React Native Metro Dev Server. + * + * More details: https://spotlightjs.com/ + * + * @default "http://localhost:8969/stream" + */ + spotlightSidecarUrl?: string; } export interface ReactNativeTransportOptions extends BrowserTransportOptions { diff --git a/test/integrations/spotlight.test.ts b/test/integrations/spotlight.test.ts new file mode 100644 index 0000000000..e58918fc06 --- /dev/null +++ b/test/integrations/spotlight.test.ts @@ -0,0 +1,97 @@ +import type { Envelope, Hub } from '@sentry/types'; +import fetchMock from 'jest-fetch-mock'; + +import { Spotlight } from '../../src/js/integrations/spotlight'; + +describe('spotlight', () => { + it('should not change the original envelope', () => { + const mockHub = createMockHub(); + + const spotlight = Spotlight(); + spotlight.setupOnce( + () => {}, + () => mockHub as unknown as Hub, + ); + + const spotlightBeforeEnvelope = mockHub.getClient().on.mock.calls[0]?.[1] as + | ((envelope: Envelope) => void) + | undefined; + + const originalEnvelopeReference = createMockEnvelope(); + spotlightBeforeEnvelope?.(originalEnvelopeReference); + + expect(spotlightBeforeEnvelope).toBeDefined(); + expect(originalEnvelopeReference).toEqual(createMockEnvelope()); + }); + + it('should remove image attachments from spotlight envelope', () => { + fetchMock.mockOnce(); + const mockHub = createMockHub(); + + const spotlight = Spotlight(); + spotlight.setupOnce( + () => {}, + () => mockHub as unknown as Hub, + ); + + const spotlightBeforeEnvelope = mockHub.getClient().on.mock.calls[0]?.[1] as + | ((envelope: Envelope) => void) + | undefined; + + spotlightBeforeEnvelope?.(createMockEnvelope()); + + expect(spotlightBeforeEnvelope).toBeDefined(); + expect(fetchMock.mock.lastCall?.[1]?.body?.toString().includes('image/png')).toBe(false); + }); +}); + +function createMockHub() { + const client = { + on: jest.fn(), + }; + + return { + getClient: jest.fn().mockReturnValue(client), + }; +} + +function createMockEnvelope(): Envelope { + return [ + { + event_id: 'event_id', + sent_at: 'sent_at', + sdk: { + name: 'sdk_name', + version: 'sdk_version', + }, + }, + [ + [ + { + type: 'event', + length: 0, + }, + { + event_id: 'event_id', + }, + ], + [ + { + type: 'attachment', + length: 10, + filename: 'filename', + }, + 'attachment', + ], + [ + { + type: 'attachment', + length: 8, + filename: 'filename2', + content_type: 'image/png', + }, + Uint8Array.from([137, 80, 78, 71, 13, 10, 26, 10]), // PNG header + ], + ], + ]; +} diff --git a/test/mockFetch.ts b/test/mockFetch.ts new file mode 100644 index 0000000000..1f06e1ce7f --- /dev/null +++ b/test/mockFetch.ts @@ -0,0 +1,2 @@ +import { enableFetchMocks } from 'jest-fetch-mock'; +enableFetchMocks(); diff --git a/test/sdk.test.ts b/test/sdk.test.ts index c65bdc7650..dd906d54d8 100644 --- a/test/sdk.test.ts +++ b/test/sdk.test.ts @@ -553,6 +553,24 @@ describe('Tests the SDK functionality', () => { ); }); + it('no spotlight integration by default', () => { + init({}); + + const actualOptions = mockedInitAndBind.mock.calls[0][secondArg] as ReactNativeClientOptions; + const actualIntegrations = actualOptions.integrations; + expect(actualIntegrations).toEqual(expect.not.arrayContaining([expect.objectContaining({ name: 'Spotlight' })])); + }); + + it('adds spotlight integration', () => { + init({ + enableSpotlight: true, + }); + + const actualOptions = mockedInitAndBind.mock.calls[0][secondArg] as ReactNativeClientOptions; + const actualIntegrations = actualOptions.integrations; + expect(actualIntegrations).toEqual(expect.arrayContaining([expect.objectContaining({ name: 'Spotlight' })])); + }); + it('no default integrations', () => { init({ defaultIntegrations: false, diff --git a/yarn.lock b/yarn.lock index e37d37f4b6..1a994deb2a 100644 --- a/yarn.lock +++ b/yarn.lock @@ -5908,7 +5908,7 @@ cosmiconfig@^5.0.5, cosmiconfig@^5.1.0: js-yaml "^3.13.1" parse-json "^4.0.0" -cross-fetch@^3.1.5: +cross-fetch@^3.0.4, cross-fetch@^3.1.5: version "3.1.8" resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.1.8.tgz#0327eba65fd68a7d119f8fb2bf9334a1a7956f82" integrity sha512-cvA+JwZoU0Xq+h6WkMvAUqPEYy92Obet6UdKLfW60qn99ftItKjB5T+BkyWOFWe2pUyfQ+IJHmpOTznqk1M6Kg== @@ -8997,6 +8997,14 @@ jest-expo@~50.0.0-alpha.0: react-test-renderer "18.2.0" stacktrace-js "^2.0.2" +jest-fetch-mock@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/jest-fetch-mock/-/jest-fetch-mock-3.0.3.tgz#31749c456ae27b8919d69824f1c2bd85fe0a1f3b" + integrity sha512-Ux1nWprtLrdrH4XwE7O7InRY6psIi3GOsqNESJgMJ+M5cv4A8Lh7SN9d2V2kKRZ8ebAfcd1LNyZguAOb6JiDqw== + dependencies: + cross-fetch "^3.0.4" + promise-polyfill "^8.1.3" + jest-get-type@^29.2.0: version "29.2.0" resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-29.2.0.tgz#726646f927ef61d583a3b3adb1ab13f3a5036408" @@ -11539,6 +11547,11 @@ promise-inflight@^1.0.1: resolved "https://registry.yarnpkg.com/promise-inflight/-/promise-inflight-1.0.1.tgz#98472870bf228132fcbdd868129bad12c3c029e3" integrity sha512-6zWPyEOFaQBJYcGMHBKTKJ3u6TBsnMFOIZSa6ce1e/ZrrsOlnHRHbabMjLiBYKp+n44X9eUI6VUPaukCXHuG4g== +promise-polyfill@^8.1.3: + version "8.3.0" + resolved "https://registry.yarnpkg.com/promise-polyfill/-/promise-polyfill-8.3.0.tgz#9284810268138d103807b11f4e23d5e945a4db63" + integrity sha512-H5oELycFml5yto/atYqmjyigJoAo3+OXwolYiH7OfQuYlAqhxNvTfiNMbV9hsC6Yp83yE5r2KTVmtrG6R9i6Pg== + promise@^7.1.1: version "7.3.1" resolved "https://registry.yarnpkg.com/promise/-/promise-7.3.1.tgz#064b72602b18f90f29192b8b1bc418ffd1ebd3bf"