From 515ec54d691cf644f726352c4a93bfbde66a9329 Mon Sep 17 00:00:00 2001 From: Krystof Woldrich Date: Thu, 16 Nov 2023 13:57:25 +0100 Subject: [PATCH 1/2] feat(rn): Use bundled Xcode scripts to create Sentry build phases --- src/react-native/react-native-wizard.ts | 58 +++++++++++++++-- src/react-native/xcode.ts | 82 +++++++++++++++++++++---- test/react-native/xcode.test.ts | 79 +++++++++++++++++++++++- 3 files changed, 199 insertions(+), 20 deletions(-) diff --git a/src/react-native/react-native-wizard.ts b/src/react-native/react-native-wizard.ts index d0fec705..c5846d66 100644 --- a/src/react-native/react-native-wizard.ts +++ b/src/react-native/react-native-wizard.ts @@ -26,8 +26,11 @@ import { findBundlePhase, patchBundlePhase, findDebugFilesUploadPhase, - addDebugFilesUploadPhase, + addDebugFilesUploadPhaseWithCli, writeXcodeProject, + addSentryWithCliToBundleShellScript, + addSentryWithBundledScriptsToBundleShellScript, + addDebugFilesUploadPhaseWithBundledScripts, } from './xcode'; import { doesAppBuildGradleIncludeRNSentryGradlePlugin, @@ -45,6 +48,7 @@ import { } from './javascript'; import { traceStep, withTelemetry } from '../telemetry'; import * as Sentry from '@sentry/node'; +import { fulfillsVersionRange } from '../utils/semver'; // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment const xcode = require('xcode'); @@ -56,6 +60,10 @@ export const RN_HUMAN_NAME = 'React Native'; export const SUPPORTED_RN_RANGE = '>=0.69.0'; +// The following SDK version ship with bundled Xcode scripts +// which simplifies the Xcode Build Phases setup. +export const SDK_XCODE_SCRIPTS_SUPPORTED_SDK_RANGE = '>=5.11.0'; + export type RNCliSetupConfigContent = Pick< Required, 'authToken' | 'org' | 'project' | 'url' @@ -118,6 +126,10 @@ export async function runReactNativeWizardWithTelemetry( packageName: RN_SDK_PACKAGE, alreadyInstalled: hasPackageInstalled(RN_SDK_PACKAGE, packageJson), }); + const sdkVersion = getPackageVersion( + RN_SDK_PACKAGE, + await getPackageDotJson(), + ); await traceStep('patch-js', () => addSentryInit({ dsn: selectedProject.keys[0].dsn.public }), @@ -125,7 +137,9 @@ export async function runReactNativeWizardWithTelemetry( if (fs.existsSync('ios')) { Sentry.setTag('patch-ios', true); - await traceStep('patch-xcode-files', () => patchXcodeFiles(cliConfig)); + await traceStep('patch-xcode-files', () => + patchXcodeFiles(cliConfig, { sdkVersion }), + ); } if (fs.existsSync('android')) { @@ -233,7 +247,12 @@ ${chalk.cyan(projectsIssuesUrl)}`); return firstErrorConfirmed; } -async function patchXcodeFiles(config: RNCliSetupConfigContent) { +async function patchXcodeFiles( + config: RNCliSetupConfigContent, + context: { + sdkVersion: string | undefined; + }, +) { await addSentryCliConfig(config, { ...propertiesCliSetupConfig, name: 'source maps and iOS debug files', @@ -281,7 +300,21 @@ async function patchXcodeFiles(config: RNCliSetupConfigContent) { 'xcode-bundle-phase-status', bundlePhase ? 'found' : 'not-found', ); - patchBundlePhase(bundlePhase); + if ( + context.sdkVersion && + fulfillsVersionRange({ + version: context.sdkVersion, + acceptableVersions: SDK_XCODE_SCRIPTS_SUPPORTED_SDK_RANGE, + canBeLatest: true, + }) + ) { + patchBundlePhase( + bundlePhase, + addSentryWithBundledScriptsToBundleShellScript, + ); + } else { + patchBundlePhase(bundlePhase, addSentryWithCliToBundleShellScript); + } Sentry.setTag('xcode-bundle-phase-status', 'patched'); }); @@ -292,7 +325,22 @@ async function patchXcodeFiles(config: RNCliSetupConfigContent) { 'xcode-debug-files-upload-phase-status', debugFilesUploadPhaseExists ? 'already-exists' : undefined, ); - addDebugFilesUploadPhase(xcodeProject, { debugFilesUploadPhaseExists }); + if ( + context.sdkVersion && + fulfillsVersionRange({ + version: context.sdkVersion, + acceptableVersions: SDK_XCODE_SCRIPTS_SUPPORTED_SDK_RANGE, + canBeLatest: true, + }) + ) { + addDebugFilesUploadPhaseWithBundledScripts(xcodeProject, { + debugFilesUploadPhaseExists, + }); + } else { + addDebugFilesUploadPhaseWithCli(xcodeProject, { + debugFilesUploadPhaseExists, + }); + } Sentry.setTag('xcode-debug-files-upload-phase-status', 'added'); }); diff --git a/src/react-native/xcode.ts b/src/react-native/xcode.ts index 3b9c3dbd..662e1dd6 100644 --- a/src/react-native/xcode.ts +++ b/src/react-native/xcode.ts @@ -21,7 +21,10 @@ export function getValidExistingBuildPhases(xcodeProject: any): BuildPhaseMap { return map; } -export function patchBundlePhase(bundlePhase: BuildPhase | undefined) { +export function patchBundlePhase( + bundlePhase: BuildPhase | undefined, + patch: (script: string) => string, +) { if (!bundlePhase) { clack.log.warn( `Could not find ${chalk.cyan( @@ -42,9 +45,7 @@ export function patchBundlePhase(bundlePhase: BuildPhase | undefined) { } const script: string = JSON.parse(bundlePhase.shellScript); - bundlePhase.shellScript = JSON.stringify( - addSentryToBundleShellScript(script), - ); + bundlePhase.shellScript = JSON.stringify(patch(script)); clack.log.success( `Patched Build phase ${chalk.cyan('Bundle React Native code and images')}.`, ); @@ -60,7 +61,10 @@ export function unPatchBundlePhase(bundlePhase: BuildPhase | undefined) { return; } - if (!bundlePhase.shellScript.match(/sentry-cli\s+react-native\s+xcode/i)) { + if ( + !bundlePhase.shellScript.match(/sentry-cli\s+react-native\s+xcode/i) && + !bundlePhase.shellScript.includes('sentry-xcode.sh') + ) { clack.log.success( `Build phase ${chalk.cyan( 'Bundle React Native code and images', @@ -97,6 +101,12 @@ export function removeSentryFromBundleShellScript(script: string): string { /\.\.\/node_modules\/@sentry\/cli\/bin\/sentry-cli\s+react-native\s+xcode\s+\$REACT_NATIVE_XCODE/i, '$REACT_NATIVE_XCODE', ) + .replace( + // eslint-disable-next-line no-useless-escape + /\"\/bin\/sh.*?sentry-xcode.sh\s+\$REACT_NATIVE_XCODE/i, + // eslint-disable-next-line no-useless-escape + '"$REACT_NATIVE_XCODE', + ) ); } @@ -107,10 +117,25 @@ export function findBundlePhase(buildPhases: BuildPhaseMap) { } export function doesBundlePhaseIncludeSentry(buildPhase: BuildPhase) { - return !!buildPhase.shellScript.match(/sentry-cli\s+react-native\s+xcode/i); + const containsSentryCliRNCommand = !!buildPhase.shellScript.match( + /sentry-cli\s+react-native\s+xcode/i, + ); + const containsBundledScript = + buildPhase.shellScript.includes('sentry-xcode.sh'); + return containsSentryCliRNCommand || containsBundledScript; +} + +export function addSentryWithBundledScriptsToBundleShellScript( + script: string, +): string { + return script.replace( + '$REACT_NATIVE_XCODE', + // eslint-disable-next-line no-useless-escape + '\\"/bin/sh ../node_modules/@sentry/react-native/scripts/sentry-xcode.sh $REACT_NATIVE_XCODE\\"', + ); } -export function addSentryToBundleShellScript(script: string): string { +export function addSentryWithCliToBundleShellScript(script: string): string { return ( 'export SENTRY_PROPERTIES=sentry.properties\n' + 'export EXTRA_PACKAGER_ARGS="--sourcemap-output $DERIVED_FILE_DIR/main.jsbundle.map"\n' + @@ -124,7 +149,36 @@ export function addSentryToBundleShellScript(script: string): string { ); } -export function addDebugFilesUploadPhase( +export function addDebugFilesUploadPhaseWithBundledScripts( + // eslint-disable-next-line @typescript-eslint/no-explicit-any + xcodeProject: any, + { debugFilesUploadPhaseExists }: { debugFilesUploadPhaseExists: boolean }, +) { + if (debugFilesUploadPhaseExists) { + clack.log.warn( + `Build phase ${chalk.cyan( + 'Upload Debug Symbols to Sentry', + )} already exists.`, + ); + return; + } + + xcodeProject.addBuildPhase( + [], + 'PBXShellScriptBuildPhase', + 'Upload Debug Symbols to Sentry', + null, + { + shellPath: '/bin/sh', + shellScript: `/bin/sh ../node_modules/@sentry/react-native/scripts/sentry-xcode-debug-files.sh`, + }, + ); + clack.log.success( + `Added Build phase ${chalk.cyan('Upload Debug Symbols to Sentry')}.`, + ); +} + +export function addDebugFilesUploadPhaseWithCli( // eslint-disable-next-line @typescript-eslint/no-explicit-any xcodeProject: any, { debugFilesUploadPhaseExists }: { debugFilesUploadPhaseExists: boolean }, @@ -204,13 +258,17 @@ export function unPatchDebugFilesUploadPhase( export function findDebugFilesUploadPhase( buildPhasesMap: Record, ): [key: string, buildPhase: BuildPhase] | undefined { - return Object.entries(buildPhasesMap).find( - ([_, buildPhase]) => + return Object.entries(buildPhasesMap).find(([_, buildPhase]) => { + const containsCliDebugUpload = typeof buildPhase !== 'string' && !!buildPhase.shellScript.match( /sentry-cli\s+(upload-dsym|debug-files upload)\b/, - ), - ); + ); + const containsBundledDebugUpload = + typeof buildPhase !== 'string' && + buildPhase.shellScript.includes('sentry-xcode-debug-files.sh'); + return containsCliDebugUpload || containsBundledDebugUpload; + }); } // eslint-disable-next-line @typescript-eslint/no-explicit-any diff --git a/test/react-native/xcode.test.ts b/test/react-native/xcode.test.ts index d690a3b1..de6d2639 100644 --- a/test/react-native/xcode.test.ts +++ b/test/react-native/xcode.test.ts @@ -1,6 +1,7 @@ /* eslint-disable no-useless-escape */ import { - addSentryToBundleShellScript, + addSentryWithBundledScriptsToBundleShellScript, + addSentryWithCliToBundleShellScript, doesBundlePhaseIncludeSentry, findBundlePhase, findDebugFilesUploadPhase, @@ -8,7 +9,7 @@ import { } from '../../src/react-native/xcode'; describe('react-native xcode', () => { - describe('addSentryToBundleShellScript', () => { + describe('addSentryWithCliToBundleShellScript', () => { it('adds sentry cli to rn bundle build phase', () => { const input = `set -e @@ -30,7 +31,31 @@ REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" /bin/sh -c "$WITH_ENVIRONMENT ../node_modules/@sentry/react-native/scripts/collect-modules.sh" `; - expect(addSentryToBundleShellScript(input)).toBe(expectedOutput); + expect(addSentryWithCliToBundleShellScript(input)).toBe(expectedOutput); + }); + }); + + describe('addSentryBundledScriptsToBundleShellScript', () => { + it('adds sentry cli to rn bundle build phase', () => { + const input = `set -e + +WITH_ENVIRONMENT="../node_modules/react-native/scripts/xcode/with-environment.sh" +REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" + +/bin/sh -c "$WITH_ENVIRONMENT $REACT_NATIVE_XCODE"`; + // actual shell script looks like this: + // /bin/sh -c "$WITH_ENVIRONMENT \"$REACT_NATIVE_XCODE\"" + // but during parsing xcode library removes the quotes + const expectedOutput = `set -e + +WITH_ENVIRONMENT="../node_modules/react-native/scripts/xcode/with-environment.sh" +REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" + +/bin/sh -c "$WITH_ENVIRONMENT \\"/bin/sh ../node_modules/@sentry/react-native/scripts/sentry-xcode.sh $REACT_NATIVE_XCODE\\""`; + + expect(addSentryWithBundledScriptsToBundleShellScript(input)).toBe( + expectedOutput, + ); }); }); @@ -59,6 +84,23 @@ REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" expect(removeSentryFromBundleShellScript(input)).toBe(expectedOutput); }); + + it('removes sentry bundled scripts from rn bundle build phase', () => { + const input = `set -e + +WITH_ENVIRONMENT="../node_modules/react-native/scripts/xcode/with-environment.sh" +REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" + +/bin/sh -c "$WITH_ENVIRONMENT \"/bin/sh ../node_modules/@sentry/react-native/scripts/sentry-xcode.sh $REACT_NATIVE_XCODE\""`; + const expectedOutput = `set -e + +WITH_ENVIRONMENT="../node_modules/react-native/scripts/xcode/with-environment.sh" +REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" + +/bin/sh -c "$WITH_ENVIRONMENT \"$REACT_NATIVE_XCODE\""`; + + expect(removeSentryFromBundleShellScript(input)).toBe(expectedOutput); + }); }); describe('findBundlePhase', () => { @@ -138,6 +180,19 @@ SENTRY_CLI="sentry-cli react-native xcode" expect(doesBundlePhaseIncludeSentry(input)).toBeTruthy(); }); + it('returns true for script containing sentry bundled script', () => { + const input = { + shellScript: `set -e +WITH_ENVIRONMENT="../node_modules/react-native/scripts/xcode/with-environment.sh" +REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" +SENTRY_CLI="sentry-cli react-native xcode" + +/bin/sh -c "$WITH_ENVIRONMENT \\"/bin/sh ../node_modules/@sentry/react-native/scripts/sentry-xcode.sh $REACT_NATIVE_XCODE"\\" +`, + }; + expect(doesBundlePhaseIncludeSentry(input)).toBeTruthy(); + }); + it('returns false', () => { const input = { // note sentry-cli can be part of the script but doesn't call react native xcode script @@ -221,6 +276,24 @@ sentry-cli upload-dsym path/to/dsym --include-sources expect(findDebugFilesUploadPhase(input)).toEqual(expected); }); + it('returns debug files build phase using bundled scripts', () => { + const input = { + 1: { + shellScript: 'foo', + }, + 2: { + shellScript: `/bin/sh ../node_modules/@sentry/react-native/scripts/sentry-xcode-debug-files.sh`, + }, + }; + const expected = [ + '2', + { + shellScript: `/bin/sh ../node_modules/@sentry/react-native/scripts/sentry-xcode-debug-files.sh`, + }, + ]; + expect(findDebugFilesUploadPhase(input)).toEqual(expected); + }); + it('returns undefined if build phase not present', () => { const input = { 1: { From be6e23c6579de62fbcd144c3d2cf1a2261e4317e Mon Sep 17 00:00:00 2001 From: Krystof Woldrich Date: Thu, 16 Nov 2023 13:58:49 +0100 Subject: [PATCH 2/2] Add changelog --- CHANGELOG.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 6b0ee254..0c2afceb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,9 @@ # Changelog +## Unreleased + +- feat(reactnative): Use Xcode scripts bundled with Sentry RN SDK (#499) + ## 3.16.4 - feat(nextjs): Add instructions for custom \_error page (#496)