diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS index 4d7c49b5fc..fd1550a503 100644 --- a/.github/CODEOWNERS +++ b/.github/CODEOWNERS @@ -1 +1 @@ -* @krystofwoldrich @lucas-zimerman @antonis +* @alwx @antonis @lucas-zimerman diff --git a/.github/ISSUE_TEMPLATE/BUG_REPORT.yml b/.github/ISSUE_TEMPLATE/BUG_REPORT.yml index adca5e3167..33d99c158b 100644 --- a/.github/ISSUE_TEMPLATE/BUG_REPORT.yml +++ b/.github/ISSUE_TEMPLATE/BUG_REPORT.yml @@ -1,7 +1,6 @@ name: '🐞 Bug Report' description: "Tell us about something that's not working the way we (probably) intend." -labels: ['Platform: React-Native', 'Type: 🪲 Bug'] -type: Bug +labels: ['React-Native', 'Bug'] body: - type: dropdown id: environment @@ -53,11 +52,11 @@ body: 'Output of the command `npx react-native@latest info` or manully describe your development environment?' value: |- ```` - ⬇ Place the `npx react-native@latest info` output here. ⬇ - - - - + ⬇ Place the `npx react-native@latest info` output here. ⬇ + + + + ```` - type: textarea diff --git a/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.yml b/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.yml index 84d067a56c..9661bd2fac 100644 --- a/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.yml +++ b/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.yml @@ -1,7 +1,6 @@ name: 💡 Feature Request description: Tell us about a problem our SDK could solve but doesn't. -labels: ['Platform: React-Native', 'enhancement'] -type: Feature +labels: ['React-Native', 'Feature'] body: - type: textarea id: problem diff --git a/.github/ISSUE_TEMPLATE/maintainer-blank.yml b/.github/ISSUE_TEMPLATE/maintainer-blank.yml index c4d42eb7df..3b8db3e22a 100644 --- a/.github/ISSUE_TEMPLATE/maintainer-blank.yml +++ b/.github/ISSUE_TEMPLATE/maintainer-blank.yml @@ -1,6 +1,6 @@ name: Blank Issue description: Blank Issue. Reserved for maintainers. -labels: ["Platform: React-Native"] +labels: ['React-Native'] body: - type: textarea id: description diff --git a/.github/workflows/add-platform-label.yml b/.github/workflows/add-platform-label.yml index 91e2cf7c4e..1846933dc7 100644 --- a/.github/workflows/add-platform-label.yml +++ b/.github/workflows/add-platform-label.yml @@ -13,5 +13,5 @@ jobs: steps: - uses: andymckay/labeler@e6c4322d0397f3240f0e7e30a33b5c5df2d39e90 # pin@1.0.4 with: - add-labels: 'Platform: React-Native' + add-labels: 'React-Native' repo-token: ${{ secrets.GITHUB_TOKEN }} diff --git a/.github/workflows/buildandtest.yml b/.github/workflows/buildandtest.yml index 7da3d56afc..c6c335fce0 100644 --- a/.github/workflows/buildandtest.yml +++ b/.github/workflows/buildandtest.yml @@ -22,10 +22,11 @@ jobs: needs: [diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock @@ -40,15 +41,43 @@ jobs: needs: [diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock - name: Install Dependencies run: yarn install + + # Default of ubuntu and apt packages are too old compared to macos packages. + # This is required for using a newer version of clang-format. + - name: Setup clang-format V20 + run: | + sudo bash -c "$(wget -O - https://apt.llvm.org/llvm.sh)" 20 + sudo apt-get install -y clang-20 clang-format-20 lld-20 lldb-20 + + - name: Set clang-format V20 as default + run: | + sudo update-alternatives --install /usr/bin/clang clang /usr/bin/clang-20 200 + sudo update-alternatives --install /usr/bin/clang++ clang++ /usr/bin/clang++-20 200 + sudo update-alternatives --install /usr/bin/clang-format clang-format /usr/bin/clang-format-20 200 + clang --version + clang-format --version + + - name: Install Swiftly + run: | + SWIFTLY_FILE="swiftly-$(uname -m).tar.gz" + curl -sL https://download.swift.org/swiftly/linux/swiftly-x86_64.tar.gz -o $SWIFTLY_FILE + tar zxf $SWIFTLY_FILE + + ./swiftly init --quiet-shell-followup + . "${SWIFTLY_HOME_DIR:-$HOME/.local/share/swiftly}/env.sh" + hash -r + sudo apt-get -y install libcurl4-openssl-dev + - name: Lint run: yarn lint @@ -58,10 +87,11 @@ jobs: needs: [diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock @@ -81,10 +111,11 @@ jobs: needs: [diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock @@ -124,22 +155,23 @@ jobs: env: YARN_ENABLE_IMMUTABLE_INSTALLS: false steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock - name: Install Dependencies run: yarn install - name: Download dist - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v5 with: name: dist path: packages/core/dist - name: Download ts3.8 - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v5 with: name: ts3.8 path: packages/core/ts3.8 @@ -154,22 +186,23 @@ jobs: needs: [job_build, diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock - name: Install Dependencies run: yarn install - name: Download dist - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v5 with: name: dist path: packages/core/dist - name: Download Expo Plugin - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v5 with: name: expo-plugin path: packages/core/plugin/build @@ -188,10 +221,11 @@ jobs: platform: ['ios', 'android'] dev: [true, false] steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock diff --git a/.github/workflows/changes-in-high-risk-code.yml b/.github/workflows/changes-in-high-risk-code.yml index 64decbe48f..e9c436ea25 100644 --- a/.github/workflows/changes-in-high-risk-code.yml +++ b/.github/workflows/changes-in-high-risk-code.yml @@ -16,7 +16,7 @@ jobs: high_risk_code: ${{ steps.changes.outputs.high_risk_code }} high_risk_code_files: ${{ steps.changes.outputs.high_risk_code_files }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - name: Get changed files id: changes uses: dorny/paths-filter@de90cc6fb38fc0963ad72b210f1f284cd68cea36 # v3.0.2 @@ -34,7 +34,7 @@ jobs: runs-on: ubuntu-latest steps: - name: Comment on PR to notify of changes in high risk files - uses: actions/github-script@v7 + uses: actions/github-script@v8 env: high_risk_code: ${{ needs.files-changed.outputs.high_risk_code_files }} with: diff --git a/.github/workflows/codegen.yml b/.github/workflows/codegen.yml index 9520480de2..855e168bab 100644 --- a/.github/workflows/codegen.yml +++ b/.github/workflows/codegen.yml @@ -36,14 +36,15 @@ jobs: --outputPath codegen \ --targetPlatform ios steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock - - uses: actions/setup-java@v4 + - uses: actions/setup-java@v5 with: java-version: '17' distribution: "adopt" diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml index 0f0faeaf15..9c51b118c6 100644 --- a/.github/workflows/codeql-analysis.yml +++ b/.github/workflows/codeql-analysis.yml @@ -40,11 +40,11 @@ jobs: steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v5 # Initializes the CodeQL tools for scanning. - name: Initialize CodeQL - uses: github/codeql-action/init@fca7ace96b7d713c7035871441bd52efbe39e27e # pin@v3.28.19 + uses: github/codeql-action/init@16140ae1a102900babc80a33c44059580f687047 # pin@v4.30.9 with: languages: ${{ matrix.language }} # If you wish to specify custom queries, you can do so here or in a config file. @@ -55,7 +55,7 @@ jobs: # Autobuild attempts to build any compiled languages (C/C++, C#, or Java). # If this step fails, then you should remove it and run the build manually (see below) - name: Autobuild - uses: github/codeql-action/autobuild@fca7ace96b7d713c7035871441bd52efbe39e27e # pin@v3.28.19 + uses: github/codeql-action/autobuild@16140ae1a102900babc80a33c44059580f687047 # pin@v4.30.9 # ℹ️ Command-line programs to run using the OS shell. # 📚 https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions @@ -66,4 +66,4 @@ jobs: # make bootstrap # make release - name: Perform CodeQL Analysis - uses: github/codeql-action/analyze@fca7ace96b7d713c7035871441bd52efbe39e27e # pin@v3.28.19 + uses: github/codeql-action/analyze@16140ae1a102900babc80a33c44059580f687047 # pin@v4.30.9 diff --git a/.github/workflows/danger.yml b/.github/workflows/danger.yml index 000b75ff3e..09d4bcb033 100644 --- a/.github/workflows/danger.yml +++ b/.github/workflows/danger.yml @@ -6,4 +6,6 @@ on: jobs: danger: - uses: getsentry/github-workflows/.github/workflows/danger.yml@v2 + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/danger@v3 diff --git a/.github/workflows/e2e-v2.yml b/.github/workflows/e2e-v2.yml index e7a4738921..2b9b6e8caa 100644 --- a/.github/workflows/e2e-v2.yml +++ b/.github/workflows/e2e-v2.yml @@ -14,7 +14,7 @@ concurrency: env: SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }} - MAESTRO_VERSION: '1.40.3' + MAESTRO_VERSION: '2.0.6' IOS_DEVICE: 'iPhone 16' IOS_VERSION: '18.1' @@ -48,7 +48,7 @@ jobs: name: Android appPlain: performance-tests/TestAppPlain/android/app/build/outputs/apk/release/app-release.apk steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - uses: ./.github/actions/disk-cleanup if: ${{ matrix.platform == 'android' }} @@ -57,13 +57,14 @@ jobs: if: ${{ matrix.platform == 'ios' }} - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: - node-version: 18 + package-manager-cache: false + node-version: 20 cache: 'yarn' cache-dependency-path: yarn.lock - - uses: actions/setup-java@v4 + - uses: actions/setup-java@v5 with: java-version: '17' distribution: "adopt" @@ -141,7 +142,7 @@ jobs: MATCH_PASSWORD: ${{ secrets.MATCH_PASSWORD }} MATCH_USERNAME: ${{ secrets.MATCH_USERNAME }} - name: Collect apps metrics - uses: getsentry/action-app-sdk-overhead-metrics@v1 + uses: getsentry/action-app-sdk-overhead-metrics@c9eca50e02d180ee07a02952c062b2f3f545f735 with: name: ${{ matrix.name }} (${{ matrix.rn-architecture }}) config: ./performance-tests/metrics-${{ matrix.platform }}.yml @@ -163,7 +164,7 @@ jobs: strategy: fail-fast: false # keeps matrix running if one fails matrix: - rn-version: ['0.65.3', '0.79.1'] + rn-version: ['0.65.3', '0.81.0'] rn-architecture: ['legacy', 'new'] platform: ['android', 'ios'] build-type: ['production'] @@ -171,9 +172,9 @@ jobs: engine: ['hermes', 'jsc'] include: - platform: ios - rn-version: '0.79.1' + rn-version: '0.81.0' xcode-version: '16.2' - runs-on: macos-15 + runs-on: macos-14 - platform: ios rn-version: '0.65.3' xcode-version: '14.2' @@ -182,7 +183,7 @@ jobs: runs-on: ubuntu-latest exclude: # exclude JSC for new RN versions (keeping the matrix manageable) - - rn-version: '0.79.1' + - rn-version: '0.81.0' engine: 'jsc' # exclude all rn versions lower than 0.70.0 for new architecture - rn-version: '0.65.3' @@ -204,7 +205,7 @@ jobs: ios-use-frameworks: 'dynamic' steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - uses: ./.github/actions/disk-cleanup if: ${{ matrix.platform == 'android' }} @@ -228,13 +229,14 @@ jobs: if: ${{ matrix.platform == 'ios' }} - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: - node-version: 18 + package-manager-cache: false + node-version: 20 cache: 'yarn' cache-dependency-path: yarn.lock - - uses: actions/setup-java@v4 + - uses: actions/setup-java@v5 with: java-version: ${{ matrix.rn-version == '0.65.3' && '11' || '17' }} distribution: 'adopt' @@ -259,9 +261,10 @@ jobs: # The old node has to be enabled after creating the test app # to avoid issues with the old node version - run: corepack disable - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 if: ${{ matrix.rn-version == '0.65.3' }} with: + package-manager-cache: false node-version: 16 - uses: ruby/setup-ruby@v1 @@ -301,7 +304,7 @@ jobs: strategy: fail-fast: false # keeps matrix running if one fails matrix: - rn-version: ['0.65.3', '0.79.1'] + rn-version: ['0.65.3', '0.81.0'] rn-architecture: ['legacy', 'new'] platform: ['android', 'ios'] build-type: ['production'] @@ -309,11 +312,11 @@ jobs: engine: ['hermes', 'jsc'] include: - platform: ios - rn-version: '0.79.1' - runs-on: macos-15 + rn-version: '0.81.0' + runs-on: macos-14 - platform: ios rn-version: '0.65.3' - runs-on: macos-15 + runs-on: macos-14 - platform: android runs-on: ubuntu-latest exclude: @@ -323,11 +326,11 @@ jobs: # e2e test only the default combinations - rn-version: '0.65.3' engine: 'hermes' - - rn-version: '0.79.1' + - rn-version: '0.81.0' engine: 'jsc' steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - name: Install Maestro uses: dniHze/maestro-test-action@bda8a93211c86d0a05b7a4597c5ad134566fbde4 # pin@v1.0.0 @@ -347,20 +350,21 @@ jobs: - name: Download App Package if: matrix.build-type == 'production' - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v5 with: name: ${{ matrix.rn-version }}-${{ matrix.rn-architecture }}-${{ matrix.engine }}-${{ matrix.platform }}-${{ matrix.build-type }}-${{ matrix.ios-use-frameworks }}-app-package path: dev-packages/e2e-tests - name: Enable Corepack run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 20 cache: 'yarn' cache-dependency-path: yarn.lock - - uses: actions/setup-java@v4 + - uses: actions/setup-java@v5 with: java-version: '17' distribution: 'adopt' @@ -391,8 +395,7 @@ jobs: force-avd-creation: false disable-animations: true disable-spellchecker: true - target: 'aosp_atd' - channel: canary # Necessary for ATDs + target: 'google_apis' emulator-options: > -no-window -no-snapshot-save diff --git a/.github/workflows/native-tests.yml b/.github/workflows/native-tests.yml index afaa4909ff..b7c6d676b7 100644 --- a/.github/workflows/native-tests.yml +++ b/.github/workflows/native-tests.yml @@ -18,16 +18,17 @@ jobs: test-ios: name: ios - runs-on: macos-15 + runs-on: macos-14 needs: [diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - name: Enable Corepack run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock @@ -50,6 +51,7 @@ jobs: xcodebuild -workspace *.xcworkspace \ -scheme $SCHEME -configuration $CONFIGURATION \ -destination "$DESTINATION" \ + -quiet \ test test-android: @@ -58,11 +60,11 @@ jobs: needs: [diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - uses: ./.github/actions/disk-cleanup - - uses: actions/setup-java@v4 + - uses: actions/setup-java@v5 with: java-version: '17' distribution: 'adopt' diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 9f785ded04..229c8e3184 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -19,18 +19,19 @@ jobs: steps: - name: Get auth token id: token - uses: actions/create-github-app-token@df432ceedc7162793a195dd1713ff69aefc7379e # v2.0.6 + uses: actions/create-github-app-token@67018539274d69449ef7c02e8e71183d1719ab42 # v2.1.4 with: app-id: ${{ vars.SENTRY_RELEASE_BOT_CLIENT_ID }} private-key: ${{ secrets.SENTRY_RELEASE_BOT_PRIVATE_KEY }} - name: Check out current commit (${{ github.sha }}) - uses: actions/checkout@v4 + uses: actions/checkout@v5 with: token: ${{ steps.token.outputs.token }} fetch-depth: 0 - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock diff --git a/.github/workflows/sample-application-expo.yml b/.github/workflows/sample-application-expo.yml index 5830f3e5fd..093088b461 100644 --- a/.github/workflows/sample-application-expo.yml +++ b/.github/workflows/sample-application-expo.yml @@ -35,7 +35,8 @@ jobs: build-type: ['dev', 'production'] include: - platform: ios - runs-on: macos-15 + xcode-version: '16.2' + runs-on: macos-14 - platform: android runs-on: ubuntu-latest - platform: web @@ -44,12 +45,13 @@ jobs: - platform: 'android' ios-use-frameworks: 'dynamic-frameworks' steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - name: Enable Corepack run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock @@ -62,7 +64,7 @@ jobs: bundler-cache: true # runs 'bundle install' and caches installed gems automatically cache-version: 1 # cache the installed gems - - uses: actions/setup-java@v4 + - uses: actions/setup-java@v5 with: java-version: '17' distribution: 'adopt' @@ -70,6 +72,9 @@ jobs: - name: Gradle cache uses: gradle/gradle-build-action@ac2d340dc04d9e1113182899e983b5400c17cda1 # v3.5.0 + - run: sudo xcode-select -s /Applications/Xcode_${{ matrix.xcode-version }}.app/Contents/Developer + if: ${{ matrix.platform == 'ios' }} + - name: Setup Global Xcode Tools if: ${{ matrix.platform == 'ios' }} run: which xcbeautify || brew install xcbeautify diff --git a/.github/workflows/sample-application.yml b/.github/workflows/sample-application.yml index 43bbc8ad9c..75ff7e2fc4 100644 --- a/.github/workflows/sample-application.yml +++ b/.github/workflows/sample-application.yml @@ -13,7 +13,7 @@ concurrency: env: SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }} - MAESTRO_VERSION: '1.40.3' + MAESTRO_VERSION: '2.0.6' RN_SENTRY_POD_NAME: RNSentry IOS_APP_ARCHIVE_PATH: sentry-react-native-sample.app.zip ANDROID_APP_ARCHIVE_PATH: sentry-react-native-sample.apk.zip @@ -43,7 +43,8 @@ jobs: build-type: ['dev', 'production'] include: - platform: ios - runs-on: macos-15 + xcode-version: '16.2' + runs-on: macos-14 - platform: macos runs-on: macos-15 - platform: android @@ -58,12 +59,13 @@ jobs: - ios-use-frameworks: 'dynamic-frameworks' platform: 'macos' steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - name: Enable Corepack run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock @@ -76,7 +78,7 @@ jobs: bundler-cache: true # runs 'bundle install' and caches installed gems automatically cache-version: 1 # cache the installed gems - - uses: actions/setup-java@v4 + - uses: actions/setup-java@v5 with: java-version: '17' distribution: 'adopt' @@ -84,6 +86,9 @@ jobs: - name: Gradle cache uses: gradle/gradle-build-action@v3 + - run: sudo xcode-select -s /Applications/Xcode_${{ matrix.xcode-version }}.app/Contents/Developer + if: ${{ matrix.platform == 'ios' }} + - name: Setup Global Xcode Tools if: ${{ matrix.platform == 'ios' }} run: which xcbeautify || brew install xcbeautify @@ -327,7 +332,7 @@ jobs: matrix: include: - platform: ios - runs-on: macos-15 + runs-on: macos-14 rn-architecture: 'new' ios-use-frameworks: 'no-frameworks' build-type: 'production' @@ -338,7 +343,7 @@ jobs: build-type: 'production' steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - name: Install Maestro uses: dniHze/maestro-test-action@bda8a93211c86d0a05b7a4597c5ad134566fbde4 # pin@v1.0.0 @@ -347,14 +352,14 @@ jobs: - name: Download iOS App Archive if: ${{ matrix.platform == 'ios' }} - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v5 with: name: sample-rn-${{ matrix.rn-architecture }}-${{ matrix.build-type }}-${{ matrix.ios-use-frameworks}}-${{ matrix.platform }} path: ${{ env.REACT_NATIVE_SAMPLE_PATH }} - name: Download Android APK if: ${{ matrix.platform == 'android' }} - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v5 with: name: sample-rn-${{ matrix.rn-architecture }}-${{ matrix.build-type }}-${{ matrix.platform }} path: ${{ env.REACT_NATIVE_SAMPLE_PATH }} @@ -373,8 +378,9 @@ jobs: - name: Enable Corepack run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock diff --git a/.github/workflows/skip-ci.yml b/.github/workflows/skip-ci.yml index 8640037002..93af26ae70 100644 --- a/.github/workflows/skip-ci.yml +++ b/.github/workflows/skip-ci.yml @@ -10,6 +10,10 @@ jobs: diff_check: runs-on: ubuntu-latest continue-on-error: true + env: + PR_NUMBER: ${{ github.event.pull_request.number }} + BASE_REF: ${{ github.base_ref }} + HEAD_REF: ${{ github.head_ref }} outputs: skip_ci: ${{ steps.check_diff.outputs.skip_ci }} @@ -17,7 +21,7 @@ jobs: - name: Check if is PR id: check-pr run: | - if [ -z "${{ github.event.pull_request.number }}" ] || [ -z "${{ github.base_ref }}" ] || [ -z "${{ github.head_ref }}" ]; then + if [ -z "$PR_NUMBER" ] || [ -z "$BASE_REF" ] || [ -z "$HEAD_REF" ]; then echo "This action is intended to be run on pull requests only." echo "is-pr=false" >> $GITHUB_OUTPUT else @@ -26,7 +30,7 @@ jobs: - name: Checkout PR Base Branch if: steps.check-pr.outputs.is-pr == 'true' - uses: actions/checkout@v4 + uses: actions/checkout@v5 with: fetch-depth: 0 ref: ${{ github.base_ref }} @@ -34,8 +38,8 @@ jobs: - name: Checkout PR Head Branch if: steps.check-pr.outputs.is-pr == 'true' run: | - git fetch origin pull/${{ github.event.pull_request.number }}/head:${{ github.head_ref }} - git checkout ${{ github.head_ref }} + git fetch origin "pull/$PR_NUMBER/head:$HEAD_REF" + git checkout "$HEAD_REF" - name: Check diff from Pull Request if: steps.check-pr.outputs.is-pr == 'true' @@ -43,7 +47,7 @@ jobs: run: | skipList=(".github/CODEOWNERS" ".prettierignore") # Ignores changelog.md, readme.md,... - fileChangesArray=($(git diff --name-only ${{ github.base_ref }}...${{ github.head_ref }} | grep -v '\.md$' || true)) + fileChangesArray=($(git diff --name-only "$BASE_REF...$HEAD_REF" | grep -v '\.md$' || true)) printf '%s\n' "${fileChangesArray[@]}" for item in "${fileChangesArray[@]}" do diff --git a/.github/workflows/testflight.yml b/.github/workflows/testflight.yml index 06bba8da11..b68637c6f2 100644 --- a/.github/workflows/testflight.yml +++ b/.github/workflows/testflight.yml @@ -14,11 +14,11 @@ jobs: upload_to_testflight: name: Build and Upload React Native Sample to Testflight - runs-on: macos-15 + runs-on: macos-14 needs: [diff_check] if: ${{ needs.diff_check.outputs.skip_ci != 'true' }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v5 - run: sudo xcode-select -s /Applications/Xcode_16.2.app/Contents/Developer - uses: ruby/setup-ruby@v1 with: @@ -27,8 +27,9 @@ jobs: bundler-cache: true # runs 'bundle install' and caches installed gems automatically cache-version: 1 # cache the installed gems - run: npm i -g corepack - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v5 with: + package-manager-cache: false node-version: 18 cache: 'yarn' cache-dependency-path: yarn.lock diff --git a/.github/workflows/update-deps.yml b/.github/workflows/update-deps.yml index 8bdad428d5..ec4199a2fd 100644 --- a/.github/workflows/update-deps.yml +++ b/.github/workflows/update-deps.yml @@ -11,78 +11,98 @@ on: jobs: android: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-android.sh - name: Android SDK - pr-strategy: update - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-android.sh + name: Android SDK + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} + + android-stubs: + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-android-stubs.sh + name: Android SDK Stubs + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} cocoa: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-cocoa.sh - name: Cocoa SDK - pr-strategy: update - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-cocoa.sh + name: Cocoa SDK + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} javascript: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-javascript.sh - name: JavaScript SDK - pr-strategy: update - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-javascript.sh + name: JavaScript SDK + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} wizard: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-wizard.sh - name: Wizard - pr-strategy: update - changelog-entry: false - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-wizard.sh + name: Wizard + changelog-entry: false + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} cli: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-cli.sh - name: CLI - pr-strategy: update - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-cli.sh + name: CLI + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} bundler-plugins: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-bundler-plugins.sh - name: Bundler Plugins - pr-strategy: update - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-bundler-plugins.sh + name: Bundler Plugins + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} sample-rn: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-rn.sh - name: React Native - pattern: '^v[0-9.]+$' # only match non-preview versions, also ignores "latest" - pr-strategy: update - changelog-entry: false - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-rn.sh + name: React Native + pattern: '^v[0-9.]+$' # only match non-preview versions, also ignores "latest" + changelog-entry: false + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} maestro: - uses: getsentry/github-workflows/.github/workflows/updater.yml@v2 - with: - path: scripts/update-maestro.sh - name: Maestro - pattern: '^v[0-9.]+$' # only match non-preview versions - pr-strategy: update - changelog-entry: false - secrets: - api-token: ${{ secrets.CI_DEPLOY_KEY }} + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-maestro.sh + name: Maestro + pattern: '^v[0-9.]+$' # only match non-preview versions + changelog-entry: false + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} + + sentry-android-gradle-plugin: + runs-on: ubuntu-latest + steps: + - uses: getsentry/github-workflows/updater@v3 + with: + path: scripts/update-sentry-android-gradle-plugin.sh + name: Sentry Android Gradle Plugin + pattern: '^[0-9.]+$' + changelog-entry: false + ssh-key: ${{ secrets.CI_DEPLOY_KEY }} diff --git a/.gitignore b/.gitignore index 1870007b3f..0467cca1e1 100644 --- a/.gitignore +++ b/.gitignore @@ -91,3 +91,6 @@ node_modules.bak # Sentry React Native Monorepo /packages/core/README.md .env.sentry-build-plugin + +# SwiftLint +swiftlint/* diff --git a/CHANGELOG.md b/CHANGELOG.md index 5e78227557..3470200ea6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,10 +2,762 @@ > [!IMPORTANT] -> If you are upgrading to the `6.x` versions of the Sentry React Native SDK from `5.x` or below, +> If you are upgrading to the `7.x` versions of the Sentry React Native SDK from `6.x` or below, > make sure you follow our [migration guide](https://docs.sentry.io/platforms/react-native/migration/) first. +## 7.4.0 + +### Features + +- Adds Console logs as Sentry Logs. ([#5261](https://github.com/getsentry/sentry-react-native/pull/5261)) +- Adds support for `propagateTraceparent` ([#5277](https://github.com/getsentry/sentry-react-native/pull/5227)) + +### Fixes + +- Fix compatibility with `react-native-legal` ([#5253](https://github.com/getsentry/sentry-react-native/pull/5253)) + - The licenses json file is correctly generated and placed into the `res/` folder now +- Handle missing shouldAddToIgnoreList callback in Metro ([#5260](https://github.com/getsentry/sentry-react-native/pull/5260)) +- Overrides the default Cocoa SDK behavior that disables Session Replay on iOS 26.0 ([#5268](https://github.com/getsentry/sentry-react-native/pull/5268)) + - If you are using Apple's Liquid Glass we recommend that you disable Session Replay on iOS to prevent potential PII leaks (see [sentry-cocoa 8.57.0 release note warning](https://github.com/getsentry/sentry-cocoa/releases/tag/8.57.0)) + +### Dependencies + +- Bump JavaScript SDK from v10.18.0 to v10.20.0 ([#5254](https://github.com/getsentry/sentry-react-native/pull/5254), [#5272](https://github.com/getsentry/sentry-react-native/pull/5272)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/develop/CHANGELOG.md#10200) + - [diff](https://github.com/getsentry/sentry-javascript/compare/10.18.0...10.20.0) +- Bump CLI from v2.56.0 to v2.56.1 ([#5257](https://github.com/getsentry/sentry-react-native/pull/5257)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2561) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.56.0...2.56.1) +- Bump Bundler Plugins from v4.3.0 to v4.4.0 ([#5256](https://github.com/getsentry/sentry-react-native/pull/5256)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#440) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/4.3.0...4.4.0) +- Bump Cocoa SDK from v8.56.2 to v8.57.0 ([#5263](https://github.com/getsentry/sentry-react-native/pull/5263)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8570) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.56.2...8.57.0) + +## 7.3.0 + +### Features + +- Adds support for Gradle 9 ([#5233](https://github.com/getsentry/sentry-react-native/pull/5233)) + +### Fixes + +- Updates `sentry-xcode.sh` and the default settings for the `project.pbxproj` to fix the issue with escape patterns in Xcode that leaded to errors during "Bundle React Native code and images" stage ([#5221](https://github.com/getsentry/sentry-react-native/pull/5221)) +- Fixes .env file loading in Expo sourcemap uploads ([#5210](https://github.com/getsentry/sentry-react-native/pull/5210)) +- Fixes the issue with changing immutable metadata structure in the contructor of `ReactNativeClient`. This structure is getting re-created instead of being modified to ensure IP address is only inferred by Relay if `sendDefaultPii` is `true` ([#5202](https://github.com/getsentry/sentry-react-native/pull/5202)) +- Removes usage of deprecated `SafeAreaView` ([#5241](https://github.com/getsentry/sentry-react-native/pull/5241)) +- Fixes session replay recording for uncaught errors ([#5243](https://github.com/getsentry/sentry-react-native/pull/5243)) +- Fixes TypeScript errors when using custom Metro configurations with Expo SDK 54 ([#5246](https://github.com/getsentry/sentry-react-native/pull/5246)) + +### Dependencies + +- Bump Cocoa SDK from v8.56.1 to v8.56.2 ([#5214](https://github.com/getsentry/sentry-react-native/pull/5214)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8562) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.56.1...8.56.2) +- Bump Android SDK from v8.21.1 to v8.23.0 ([#5193](https://github.com/getsentry/sentry-react-native/pull/5193), [#5194](https://github.com/getsentry/sentry-react-native/pull/5194), [#5232](https://github.com/getsentry/sentry-react-native/pull/5232)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8230) + - [diff](https://github.com/getsentry/sentry-java/compare/8.21.1...8.23.0) +- Bump CLI from v2.55.0 to v2.56.0 ([#5223](https://github.com/getsentry/sentry-react-native/pull/5223)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2560) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.55.0...2.56.0) +- Bump JavaScript SDK from v10.12.0 to v10.18.0 ([#5195](https://github.com/getsentry/sentry-react-native/pull/5195), [#5245](https://github.com/getsentry/sentry-react-native/pull/5245)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/develop/CHANGELOG.md#10180) + - [diff](https://github.com/getsentry/sentry-javascript/compare/10.12.0...10.18.0) +- Bump Android SDK Stubs from v8.22.0 to v8.23.0 ([#5231](https://github.com/getsentry/sentry-react-native/pull/5231)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8230) + - [diff](https://github.com/getsentry/sentry-java/compare/8.22.0...8.23.0) + +## 7.2.0 + +### Features + +- Enable logs on native side of iOS ([#5190](https://github.com/getsentry/sentry-react-native/pull/5190)) +- Add mobile replay attributes to logs ([#5165](https://github.com/getsentry/sentry-react-native/pull/5165)) + +### Fixes + +- Vendor `metro/countLines` function to avoid issues with the private import ([#5185](https://github.com/getsentry/sentry-react-native/pull/5185)) +- Fix baseJSBundle and bundleToString TypeErrors with Metro 0.83.2 ([#5206](https://github.com/getsentry/sentry-react-native/pull/5206)) + +### Dependencies + +- Bump CLI from v2.53.0 to v2.55.0 ([#5179](https://github.com/getsentry/sentry-react-native/pull/5179)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2550) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.53.0...2.55.0) +- Bump Cocoa SDK from v8.56.0 to v8.56.1 ([#5197](https://github.com/getsentry/sentry-react-native/pull/5197)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8561) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.56.0...8.56.1) + +## 7.1.0 + +### Fixes + +- Session Replay: Allow excluding `sentry-android-replay` from android targets ([#5174](https://github.com/getsentry/sentry-react-native/pull/5174)) + - If you are not interested in using Session Replay, you can exclude the `sentry-android-replay` module from your Android targets as follows (saves nearly 40KB compressed and 80KB uncompressed off the bundle size): + + ```gradle + // from the android's root build.gradle file + subprojects { + configurations.all { + exclude group: 'io.sentry', module: 'sentry-android-replay' + } + } + ``` + +### Dependencies + +- Bump JavaScript SDK from v10.8.0 to v10.12.0 ([#5142](https://github.com/getsentry/sentry-react-native/pull/5142), [#5145](https://github.com/getsentry/sentry-react-native/pull/5145), [#5157](https://github.com/getsentry/sentry-react-native/pull/5157), [#5175](https://github.com/getsentry/sentry-react-native/pull/5175)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/develop/CHANGELOG.md#10120) + - [diff](https://github.com/getsentry/sentry-javascript/compare/10.8.0...10.12.0) +- Bump Cocoa SDK from v8.53.2 to v8.56.0 ([#5036](https://github.com/getsentry/sentry-react-native/pull/5036), [#5172](https://github.com/getsentry/sentry-react-native/pull/5172)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8560) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.53.2...8.56.0) +- Bump Android SDK from v8.20.0 to v8.21.1 ([#5155](https://github.com/getsentry/sentry-react-native/pull/5155)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8211) + - [diff](https://github.com/getsentry/sentry-java/compare/8.20.0...8.21.1) + +## 7.0.1 + +### Important Changes + +This release includes a fix for a [behaviour change](https://docs.sentry.io/platforms/javascript/migration/v8-to-v9/#behavior-changes) +that was originally fixed on version 6.21.0 of the React Native SDK: User IP Addresses should only be added to Sentry events automatically, +if `sendDefaultPii` was set to `true`. + +To avoid making a major bump, the fix was patched on the current version and not by bumping to V8. +There is _no API_ breakage involved and hence it is safe to update. +However, after updating the SDK, events (errors, traces, replays, etc.) sent from the browser, will only include +user IP addresses, if you set `sendDefaultPii: true` in your `Sentry.init` options. + +We apologize for any inconvenience caused! + +### Fixes + +- Ensure IP address is only inferred by Relay if `sendDefaultPii` is `true` ([#5138](https://github.com/getsentry/sentry-react-native/pull/5137)) + +### Dependencies + +- Bump Bundler Plugins from v4.2.0 to v4.3.0 ([#5131](https://github.com/getsentry/sentry-react-native/pull/5131)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#430) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/4.2.0...4.3.0) + +## 7.0.0 + +### Upgrading from 6.x to 7.0 + +Version 7 of the Sentry React Native SDK primarily introduces API cleanup and version support changes based on the Sentry Javascript SDK versions 9 and 10. This update contains behavioral changes that will not be caught by type checkers, linters, or tests, so we recommend carefully reading through the entire migration guide instead of relying on automatic tooling. + +Version 7 of the SDK is compatible with Sentry self-hosted versions 25.2.0 or higher (up from 24.4.2 for v6). Lower versions may continue to work, but may not support all features. + +See our [migration docs](https://docs.sentry.io/platforms/react-native/migration/v6-to-v7/) for more information. + +### Major Changes + +- Ensure IP address is only inferred by Relay if `sendDefaultPii` is `true` ([#5111](https://github.com/getsentry/sentry-react-native/pull/5111)) +- Set `{{auto}}` if `user.ip_address` is `undefined` and `sendDefaultPii: true` ([#4466](https://github.com/getsentry/sentry-react-native/pull/4466)) +- `Sentry.captureUserFeedback` removed, use `Sentry.captureFeedback` instead ([#4855](https://github.com/getsentry/sentry-react-native/pull/4855)) +- Exceptions from `captureConsoleIntegration` are now marked as handled: true by default +- `shutdownTimeout` moved from `core` to `@sentry/react-native` +- `hasTracingEnabled` was renamed to `hasSpansEnabled` +- You can no longer drop spans or return null on `beforeSendSpan` hook +- Tags formatting logic updated ([#4965](https://github.com/getsentry/sentry-react-native/pull/4965)) +Here are the altered/unaltered types, make sure to update your UI filters and alerts. + + Unaltered: string, null, number, and undefined values remain unchanged. + + Altered: Boolean values are now capitalized: true -> True, false -> False. + +### Removed types + +- TransactionNamingScheme +- Request +- Scope (prefer using the Scope class) + +### Other removed items. + +- `autoSessionTracking` from options. + To enable session tracking, ensure that `enableAutoSessionTracking` is enabled. +- `enableTracing`. Instead, set `tracesSampleRate` to a value greater than `zero` to `enable tracing`, `0` to keep tracing integrations active without sampling, or `undefined` to disable the performance integration. +- `getCurrentHub()`, `Hub`, and `getCurrentHubShim()` +- `spanId` from propagation `context` +- metrics API +- `transactionContext` from `samplingContext` +- `@sentry/utils` package, the exports were moved to `@sentry/core` +- Standalone `Client` interface & deprecate `BaseClient` + +### Changes + +- Expose `featureFlagsIntegration` ([#4984](https://github.com/getsentry/sentry-react-native/pull/4984)) +- Expose `logger` and `consoleLoggingIntegration` ([#4930](https://github.com/getsentry/sentry-react-native/pull/4930)) +- Remove deprecated `appOwnership` constant use in Expo Go detection ([#4893](https://github.com/getsentry/sentry-react-native/pull/4893)) +- Disable AppStart and NativeFrames in unsupported environments (web, Expo Go) ([#4897](https://github.com/getsentry/sentry-react-native/pull/4897)) +- Use `Replay` interface for `browserReplayIntegration` return type ([#4858](https://github.com/getsentry/sentry-react-native/pull/4858)) +- Allow using `browserReplayIntegration` without `isWeb` guard ([#4858](https://github.com/getsentry/sentry-react-native/pull/4858)) + - The integration returns noop in non-browser environments +- Use single `encodeUTF8` implementation through the SDK ([#4885](https://github.com/getsentry/sentry-react-native/pull/4885)) +- Use global `TextEncoder` (available with Hermes in React Native 0.74 or higher) to improve envelope encoding performance. ([#4874](https://github.com/getsentry/sentry-react-native/pull/4874)) +- `breadcrumbsIntegration` disables React Native incompatible options automatically ([#4886](https://github.com/getsentry/sentry-react-native/pull/4886)) +- Fork `scope` if custom scope is passed to `startSpanManual` or `startSpan` +- On React Native Web, `browserSessionIntegration` is added when `enableAutoSessionTracking` is set to `True` ([#4732](https://github.com/getsentry/sentry-react-native/pull/4732)) +- Change `Cold/Warm App Start` span description to `Cold/Warm Start` ([#4636](https://github.com/getsentry/sentry-react-native/pull/4636)) + +### Features + +- Add support for Log tracing ([#4827](https://github.com/getsentry/sentry-react-native/pull/4827), [#5122](https://github.com/getsentry/sentry-react-native/pull/5122)) + +To enable it add the following code to your Sentry Options: + +```js +Sentry.init({ + enableLogs: true, +}); +``` + +You can also filter the logs being collected by adding `beforeSendLogs` + +```js +Sentry.init({ + enableLogs: true, + beforeSendLog: log => { + return log; + }, +}); +``` + +- Automatically detect Release name and version for Expo Web ([#4967](https://github.com/getsentry/sentry-react-native/pull/4967)) + +### Fixes + +- Align span description with other platforms ([#4636](https://github.com/getsentry/sentry-react-native/pull/4636)) by @krystofwoldrich +- Tags with symbol are now logged ([#4965](https://github.com/getsentry/sentry-react-native/pull/4965)) +- IgnoreError now filters Native errors ([#4948](https://github.com/getsentry/sentry-react-native/pull/4948)) + +You can use strings to filter errors or RegEx for filtering with a pattern. + +example: + +```typescript + ignoreErrors: [ + '1234', // Will filter any error message that contains 1234. + '.*1234', // Will not filter as regex, instead will filter messages that contains '.*1234" + /.*1234/, // Regex will filter any error message that ends with 1234 + /.*1234.*/ // Regex will filter any error message that contains 1234. + ] +``` + +- Expo Updates Context is passed to native after native init to be available for crashes ([#4808](https://github.com/getsentry/sentry-react-native/pull/4808)) +- Expo Updates Context values should all be lowercase ([#4809](https://github.com/getsentry/sentry-react-native/pull/4809)) +- Avoid duplicate network requests (fetch, xhr) by default ([#4816](https://github.com/getsentry/sentry-react-native/pull/4816)) + - `traceFetch` is disabled by default on mobile as RN uses a polyfill which will be traced by `traceXHR` + +### Dependencies + +- Bump JavaScript SDK v10.8.0 ([#5123](https://github.com/getsentry/sentry-react-native/pull/5123)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/develop/CHANGELOG.md#1080) +- Bump Android SDK to v8.20.0 ([#5106](https://github.com/getsentry/sentry-react-native/pull/5106)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8200) +- Bump CLI to v2.53.0 ([5120](https://github.com/getsentry/sentry-react-native/pull/5120). [#4804](https://github.com/getsentry/sentry-react-native/pull/4804), [#4818](https://github.com/getsentry/sentry-react-native/pull/4818)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2530) +- Bump Bundler Plugins to v4.2.0 ([#5113](https://github.com/getsentry/sentry-react-native/pull/5113), [#4805](https://github.com/getsentry/sentry-react-native/pull/4805)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#340) +- Bump Cocoa SDK to v8.53.2 ([#4986](https://github.com/getsentry/sentry-react-native/pull/4986)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8532) + +## 7.0.0-rc.2 + +### Important Changes + +- Ensure IP address is only inferred by Relay if `sendDefaultPii` is `true` ([#5111](https://github.com/getsentry/sentry-react-native/pull/5111)) + +This release includes a fix for a [behaviour change](https://docs.sentry.io/platforms/javascript/migration/v8-to-v9/#behavior-changes) +that was originally introduced with v9 of the JavaScript SDK included in v7.0.0-beta.0: User IP Addresses should only be added to Sentry events automatically, +if `sendDefaultPii` was set to `true`. + +We apologize for any inconvenience caused! + +### Features + +- Logs now contains more attributes like release, os and device information ([#5032](https://github.com/getsentry/sentry-react-native/pull/5032)) + +### Dependencies + +- Bump Android SDK from v8.17.0 to v8.20.0 ([#5034](https://github.com/getsentry/sentry-react-native/pull/5034), [#5063](https://github.com/getsentry/sentry-react-native/pull/5063), [#5106](https://github.com/getsentry/sentry-react-native/pull/5106)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8200) + - [diff](https://github.com/getsentry/sentry-java/compare/8.17.0...8.20.0) +- Bump JavaScript SDK from v9.22.0 to v10.7.0 ([#5111](https://github.com/getsentry/sentry-react-native/pull/5111)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/develop/CHANGELOG.md#1070) + - [diff](https://github.com/getsentry/sentry-javascript/compare/9.22.0...10.7.0) + + +## 7.0.0-rc.1 + +### Various fixes & improvements + +- fix(sdk): Set `{{auto}}` if `user.ip_address` is `undefined` and `sendDefaultPii: true` ([#4466](https://github.com/getsentry/sentry-react-native/pull/4466)) by @krystofwoldrich +- fix(appStart): Align span description with other platforms ([#4636](https://github.com/getsentry/sentry-react-native/pull/4636)) by @krystofwoldrich + +### Dependencies + +- Bump Cocoa SDK from v8.53.1 to v8.53.2 ([#4986](https://github.com/getsentry/sentry-react-native/pull/4986)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8532) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.53.1...8.53.2) +- Bump CLI from v2.47.0 to v2.50.2 ([#5007](https://github.com/getsentry/sentry-react-native/pull/5007)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2502) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.47.0...2.50.2) +- Bump Bundler Plugins from v3.5.0 to v4.0.2 ([#5030](https://github.com/getsentry/sentry-react-native/pull/5030)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#402) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/3.5.0...4.0.2) + +## 7.0.0-beta.2 + +### Features + +- Automatically detect Release name and version for Expo Web ([#4967](https://github.com/getsentry/sentry-react-native/pull/4967)) + +### Changes + +- Expose `featureFlagsIntegration` ([#4984](https://github.com/getsentry/sentry-react-native/pull/4984)) + +### Breaking changes + +- Tags formatting logic updated ([#4965](https://github.com/getsentry/sentry-react-native/pull/4965)) +Here are the altered/unaltered types, make sure to update your UI filters and alerts. + + Unaltered: string, null, number, and undefined values remain unchanged. + + Altered: Boolean values are now capitalized: true -> True, false -> False. + +### Fixes + +- tags with symbol are now logged ([#4965](https://github.com/getsentry/sentry-react-native/pull/4965)) +- ignoreError now filters Native errors ([#4948](https://github.com/getsentry/sentry-react-native/pull/4948)) + +You can use strings to filter errors or RegEx for filtering with a pattern. + +example: + +```typescript + ignoreErrors: [ + '1234', // Will filter any error message that contains 1234. + '.*1234', // Will not filter as regex, instead will filter messages that contains '.*1234" + /.*1234/, // Regex will filter any error message that ends with 1234 + /.*1234.*/ // Regex will filter any error message that contains 1234. + ] +``` + +### Dependencies + +- Bump Android SDK from v8.14.0 to v8.17.0 ([#4953](https://github.com/getsentry/sentry-react-native/pull/4953), [#4955](https://github.com/getsentry/sentry-react-native/pull/4955), [#4987](https://github.com/getsentry/sentry-react-native/pull/4987)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8170) + - [diff](https://github.com/getsentry/sentry-java/compare/8.14.0...8.17.0) +- Bump Cocoa SDK from v8.52.1 to v8.53.1 ([#4950](https://github.com/getsentry/sentry-react-native/pull/4950)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8531) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.52.1...8.53.1) +- Bump CLI from v2.46.0 to v2.47.0 ([#4979](https://github.com/getsentry/sentry-react-native/pull/4979)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2470) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.46.0...2.47.0) + +## 7.0.0-beta.1 + +### Upgrading from 6.x to 7.0 + +Version 7 of the Sentry React Native SDK primarily introduces API cleanup and version support changes based on the Sentry Javascript SDK version 9. This update contains behavioral changes that will not be caught by type checkers, linters, or tests, so we recommend carefully reading through the entire migration guide instead of relying on automatic tooling. + +Version 7 of the SDK is compatible with Sentry self-hosted versions 24.4.2 or higher (unchanged from v6). Lower versions may continue to work, but may not support all features. + +### Major Changes + +- Set `{{auto}}` if `user.ip_address` is `undefined` and `sendDefaultPii: true` ([#4466](https://github.com/getsentry/sentry-react-native/pull/4466)) +- `Sentry.captureUserFeedback` removed, use `Sentry.captureFeedback` instead ([#4855](https://github.com/getsentry/sentry-react-native/pull/4855)) + +### Major Changes from Sentry JS SDK v9 + +- Exceptions from `captureConsoleIntegration` are now marked as handled: true by default +- `shutdownTimeout` moved from `core` to `@sentry/react-native` +- `hasTracingEnabled` was renamed to `hasSpansEnabled` +- You can no longer drop spans or return null on `beforeSendSpan` hook +- Fork `scope` if custom scope is passed to `startSpanManual` or `startSpan` + +#### Removed types + +- TransactionNamingScheme +- Request +- Scope (prefer using the Scope class) + +#### Other removed items. + +- `autoSessionTracking` from options. + To enable session tracking, ensure that `enableAutoSessionTracking` is enabled. +- `enableTracing`. Instead, set `tracesSampleRate` to a value greater than `zero` to `enable tracing`, `0` to keep tracing integrations active without sampling, or `undefined` to disable the performance integration. +- `getCurrentHub()`, `Hub`, and `getCurrentHubShim()` +- `spanId` from propagation `context` +- metrics API +- `transactionContext` from `samplingContext` +- `@sentry/utils` package, the exports were moved to `@sentry/core` +- Standalone `Client` interface & deprecate `BaseClient` + +### Features + +- Add experimental support for Log tracing ([#4827](https://github.com/getsentry/sentry-react-native/pull/4827)) + +To enable it add the following code to your Sentry Options: + +```typescript +Sentry.init({ + // other options... + _experiments: { + enableLogs: true, + }, +}); +``` + +You can also filter the logs being collected by adding beforeSendLogs into `_experiments` + +```typescript +Sentry.init({ + // other options... + _experiments: { + enableLogs: true, + beforeSendLog: (log) => { + return log; + }, + } +}); +``` + +### Changes + +- Expose `logger` and `consoleLoggingIntegration` ([#4930](https://github.com/getsentry/sentry-react-native/pull/4930)) +- Remove deprecated `appOwnership` constant use in Expo Go detection ([#4893](https://github.com/getsentry/sentry-react-native/pull/4893)) +- Disable AppStart and NativeFrames in unsupported environments (web, Expo Go) ([#4897](https://github.com/getsentry/sentry-react-native/pull/4897)) + +### Self Hosted + +- It is recommended to use Sentry Self Hosted version `25.2.0` or new for React Native V7 or newer + +### Dependencies + +- Bump Android SDK from v8.13.2 to v8.14.0 ([#4929](https://github.com/getsentry/sentry-react-native/pull/4929), [#4934](https://github.com/getsentry/sentry-react-native/pull/4934)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8140) + - [diff](https://github.com/getsentry/sentry-java/compare/8.13.2...8.14.0) +- Bump Cocoa SDK from v8.52.0 to v8.52.1 ([#4899](https://github.com/getsentry/sentry-react-native/pull/4899)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8521) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.52.0...8.52.1) + +## 7.0.0-beta.0 + +### Upgrading from 6.x to 7.0 + +Version 7 of the Sentry React Native SDK primarily introduces API cleanup and version support changes based on the Sentry Javascript SDK version 9. This update contains behavioral changes that will not be caught by type checkers, linters, or tests, so we recommend carefully reading through the entire migration guide instead of relying on automatic tooling. + +Version 7 of the SDK is compatible with Sentry self-hosted versions 24.4.2 or higher (unchanged from v6). Lower versions may continue to work, but may not support all features. + +### Major Changes + +- Set `{{auto}}` if `user.ip_address` is `undefined` and `sendDefaultPii: true` ([#4466](https://github.com/getsentry/sentry-react-native/pull/4466)) +- `Sentry.captureUserFeedback` removed, use `Sentry.captureFeedback` instead ([#4855](https://github.com/getsentry/sentry-react-native/pull/4855)) + +### Major Changes from Sentry JS SDK v9 + +- Exceptions from `captureConsoleIntegration` are now marked as handled: true by default +- `shutdownTimeout` moved from `core` to `@sentry/react-native` +- `hasTracingEnabled` was renamed to `hasSpansEnabled` +- You can no longer drop spans or return null on `beforeSendSpan` hook +- Fork `scope` if custom scope is passed to `startSpanManual` or `startSpan` + +#### Removed types + +- TransactionNamingScheme +- Request +- Scope (prefer using the Scope class) + +#### Other removed items. + +- `autoSessionTracking` from options. + To enable session tracking, ensure that `enableAutoSessionTracking` is enabled. +- `enableTracing`. Instead, set `tracesSampleRate` to a value greater than `zero` to `enable tracing`, `0` to keep tracing integrations active without sampling, or `undefined` to disable the performance integration. +- `getCurrentHub()`, `Hub`, and `getCurrentHubShim()` +- `spanId` from propagation `context` +- metrics API +- `transactionContext` from `samplingContext` +- `@sentry/utils` package, the exports were moved to `@sentry/core` +- Standalone `Client` interface & deprecate `BaseClient` + +### Changes + +- Use `Replay` interface for `browserReplayIntegration` return type ([#4858](https://github.com/getsentry/sentry-react-native/pull/4858)) +- Allow using `browserReplayIntegration` without `isWeb` guard ([#4858](https://github.com/getsentry/sentry-react-native/pull/4858)) + - The integration returns noop in non-browser environments +- Use single `encodeUTF8` implementation through the SDK ([#4885](https://github.com/getsentry/sentry-react-native/pull/4885)) +- Use global `TextEncoder` (available with Hermes in React Native 0.74 or higher) to improve envelope encoding performance. ([#4874](https://github.com/getsentry/sentry-react-native/pull/4874)) +- `breadcrumbsIntegration` disables React Native incompatible options automatically ([#4886](https://github.com/getsentry/sentry-react-native/pull/4886)) +- On React Native Web, `browserSessionIntegration` is added when `enableAutoSessionTracking` is set to `True` ([#4732](https://github.com/getsentry/sentry-react-native/pull/4732)) +- Change `Cold/Warm App Start` span description to `Cold/Warm Start` ([#4636](https://github.com/getsentry/sentry-react-native/pull/4636)) + +### Dependencies + +- Bump JavaScript SDK from v8.54.0 to v9.22.0 ([#4568](https://github.com/getsentry/sentry-react-native/pull/4568), [#4752](https://github.com/getsentry/sentry-react-native/pull/4752), [#4860](https://github.com/getsentry/sentry-react-native/pull/4860)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/9.22.0/CHANGELOG.md) + - [diff](https://github.com/getsentry/sentry-javascript/compare/8.54.0...9.22.0) +- Bump Android SDK from v7.20.1 to v8.13.2 ([#4490](https://github.com/getsentry/sentry-react-native/pull/4490), [#4847](https://github.com/getsentry/sentry-react-native/pull/4847)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8132) + - [diff](https://github.com/getsentry/sentry-java/compare/7.20.1...8.13.2) +- Bump Cocoa SDK from v8.50.0 to v8.52.0 ([#4887](https://github.com/getsentry/sentry-react-native/pull/4887)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8520) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.50.0...8.52.0) +- Bump CLI from v2.43.0 to v2.46.0 ([#4866]([???](https://github.com/getsentry/sentry-react-native/pull/4866))) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2460) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.43.0...2.46.0) +- Bump Bundler Plugins from v3.4.0 to v3.5.0 ([#4850](https://github.com/getsentry/sentry-react-native/pull/4850)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#350) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/3.4.0...3.5.0) + +## 7.0.0-alpha.0 + +### Upgrading from 6.x to 7.0 + +Version 7 of the Sentry React Native SDK primarily introduces API cleanup and version support changes based on the Sentry Javascript SDK version 9. This update contains behavioral changes that will not be caught by type checkers, linters, or tests, so we recommend carefully reading through the entire migration guide instead of relying on automatic tooling. + +Version 7 of the SDK is compatible with Sentry self-hosted versions 24.4.2 or higher (unchanged from v6). Lower versions may continue to work, but may not support all features. + +### Fixes + +- Expo Updates Context is passed to native after native init to be available for crashes ([#4808](https://github.com/getsentry/sentry-react-native/pull/4808)) +- Expo Updates Context values should all be lowercase ([#4809](https://github.com/getsentry/sentry-react-native/pull/4809)) +- Avoid duplicate network requests (fetch, xhr) by default ([#4816](https://github.com/getsentry/sentry-react-native/pull/4816)) + - `traceFetch` is disabled by default on mobile as RN uses a polyfill which will be traced by `traceXHR` + +### Major Changes + +- Set `{{auto}}` if `user.ip_address` is `undefined` and `sendDefaultPii: true` ([#4466](https://github.com/getsentry/sentry-react-native/pull/4466)) +- Exceptions from `captureConsoleIntegration` are now marked as handled: true by default +- `shutdownTimeout` moved from `core` to `@sentry/react-native` +- `hasTracingEnabled` was renamed to `hasSpansEnabled` +- You can no longer drop spans or return null on `beforeSendSpan` hook + +### Removed types + +- TransactionNamingScheme +- Request +- Scope (prefer using the Scope class) + +### Other removed items. + +- `autoSessionTracking` from options. + To enable session tracking, ensure that `enableAutoSessionTracking` is enabled. +- `enableTracing`. Instead, set `tracesSampleRate` to a value greater than `zero` to `enable tracing`, `0` to keep tracing integrations active without sampling, or `undefined` to disable the performance integration. +- `getCurrentHub()`, `Hub`, and `getCurrentHubShim()` +- `spanId` from propagation `context` +- metrics API +- `transactionContext` from `samplingContext` +- `@sentry/utils` package, the exports were moved to `@sentry/core` +- Standalone `Client` interface & deprecate `BaseClient` + +### Other Changes + +- Fork `scope` if custom scope is passed to `startSpanManual` or `startSpan` +- On React Native Web, `browserSessionIntegration` is added when `enableAutoSessionTracking` is set to `True` ([#4732](https://github.com/getsentry/sentry-react-native/pull/4732)) +- Change `Cold/Warm App Start` span description to `Cold/Warm Start` ([#4636](https://github.com/getsentry/sentry-react-native/pull/4636)) + +### Dependencies + +- Bump JavaScript SDK from v8.54.0 to v9.12.0 ([#4568](https://github.com/getsentry/sentry-react-native/pull/4568), [#4752](https://github.com/getsentry/sentry-react-native/pull/4752)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/9.12.0/CHANGELOG.md) + - [diff](https://github.com/getsentry/sentry-javascript/compare/8.54.0...9.12.0) +- Bump Android SDK from v7.20.1 to v8.11.1 ([#4490](https://github.com/getsentry/sentry-react-native/pull/4490)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#8111) + - [diff](https://github.com/getsentry/sentry-java/compare/7.20.1...8.11.1) +- Bump Bundler Plugins from v3.3.1 to v3.4.0 ([#4805](https://github.com/getsentry/sentry-react-native/pull/4805)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#340) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/3.3.1...3.4.0) +- Bump Cocoa SDK from v8.49.2 to v8.50.0 ([#4807](https://github.com/getsentry/sentry-react-native/pull/4807)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8500) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.49.2...8.50.0) + +## 6.21.0 + +### Important Changes + +- **fix(browser): Ensure IP address is only inferred by Relay if `sendDefaultPii` is `true`** ([#5092](https://github.com/getsentry/sentry-react-native/pull/5092)) + +This release includes a fix for a [behaviour change](https://docs.sentry.io/platforms/javascript/migration/v8-to-v9/#behavior-changes) +that was originally introduced with v9 of the JavaScript SDK: User IP Addresses should only be added to Sentry events automatically, +if `sendDefaultPii` was set to `true`. + +However, the change in v9 required further internal adjustment, which should have been included in v10 of the SDK. +To avoid making a major bump, the fix was patched on the current version and not by bumping to V10. +There is _no API_ breakage involved and hence it is safe to update. +However, after updating the SDK, events (errors, traces, replays, etc.) sent from the browser, will only include +user IP addresses, if you set `sendDefaultPii: true` in your `Sentry.init` options. + +We apologize for any inconvenience caused! + +### Fixes + +- Fix Expo prebuild failed on cached builds ([#5098](https://github.com/getsentry/sentry-react-native/pull/5098)) +- Remove the warning that used to indicate that Time To Initial Display and Time To Full Display are not supported ([#5081](https://github.com/getsentry/sentry-react-native/pull/5081)) + +### Dependencies + +- Bump CLI from v2.51.1 to v2.53.0 ([#5075](https://github.com/getsentry/sentry-react-native/pull/5075), [#5120](https://github.com/getsentry/sentry-react-native/pull/5120)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2530) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.51.1...2.53.0) +- Bump Bundler Plugins from v4.1.1 to v4.2.0 ([#5113](https://github.com/getsentry/sentry-react-native/pull/5113)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#420) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/4.1.1...4.2.0) + +## 6.20.0 + +### Features + +- Support for React Native 0.81 ([#5051](https://github.com/getsentry/sentry-react-native/pull/5051)) +- Support New Hermes Runtime Access Pattern ([#5051](https://github.com/getsentry/sentry-react-native/pull/5051)) +- Support Metro 0.83 ([#5035](https://github.com/getsentry/sentry-react-native/pull/5035)) + +### Fixes + +- Correct detection of whether turbo modules are available ([#5064](https://github.com/getsentry/sentry-react-native/pull/5064)) + +### Dependencies + +- Bump CLI from v2.50.2 to v2.51.1 ([#5053](https://github.com/getsentry/sentry-react-native/pull/5053), [#5058](https://github.com/getsentry/sentry-react-native/pull/5058)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2511) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.50.2...2.51.1) +- Bump Bundler Plugins from v4.0.2 to v4.1.1 ([#5062](https://github.com/getsentry/sentry-react-native/pull/5062), [#5072](https://github.com/getsentry/sentry-react-native/pull/5072)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#411) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/4.0.2...4.1.1) + +## 6.19.0 + +### Fixes + +- Warnings when .env.sentry-build-plugin is not set on Sentry/CLI ([#5029](https://github.com/getsentry/sentry-react-native/pull/5029)) +- Fix for `sentry-cli` path discovery not working on Android ([#5009](https://github.com/getsentry/sentry-react-native/pull/5009)) +- Export `addIntegration` from `@sentry/core` ([#5020](https://github.com/getsentry/sentry-react-native/pull/5020)) + +### Features + +- Adds `replaysSessionQuality` Session Replay option to control replay quality and performance overhead on mobile ([#5001](https://github.com/getsentry/sentry-react-native/pull/5001)) + + ```js + import * as Sentry from '@sentry/react-native'; + + Sentry.init({ + replaysSessionSampleRate: 1.0, + replaysSessionQuality: 'low', // possible values: low, medium (default), high + integrations: [Sentry.mobileReplayIntegration()], + }); + ``` + +### Dependencies + +- Bump CLI from v2.50.0 to v2.50.2 ([#5007](https://github.com/getsentry/sentry-react-native/pull/5007)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2502) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.50.0...2.50.2) +- Bump Bundler Plugins from v3.6.1 to v4.0.2 ([#5000](https://github.com/getsentry/sentry-react-native/pull/5000), [#5021](https://github.com/getsentry/sentry-react-native/pull/5021), [#5030](https://github.com/getsentry/sentry-react-native/pull/5030)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#402) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/3.6.1...4.0.2) + +## 6.18.1 + +### Fixes + +- Fixed Sentry CLI executable path resolution that was causing iOS build script failures ([#5003](https://github.com/getsentry/sentry-react-native/pull/5003)) + +## 6.18.0 + +> [!WARNING] +> This release contains an issue where Sentry-CLI may not be found on iOS builds if not defined by environment variable. +> See PR [#5003](github.com/getsentry/sentry-react-native/pull/5003) for more details. + +### Fixes + +- SDK now Builds when using PnPM ([#4977](https://github.com/getsentry/sentry-react-native/pull/4977)) +- Skip idle span creation when app is in background ([#4995](https://github.com/getsentry/sentry-react-native/pull/4995)) + +### Dependencies + +- Bump JavaScript SDK from v8.54.0 to v8.55.0 ([#4981](https://github.com/getsentry/sentry-react-native/pull/4981)) + - [changelog](https://github.com/getsentry/sentry-javascript/blob/develop/CHANGELOG.md#8550) + - [diff](https://github.com/getsentry/sentry-javascript/compare/8.54.0...8.55.0) +- Bump Cocoa SDK from v8.53.1 to v8.53.2 ([#4986](https://github.com/getsentry/sentry-react-native/pull/4986)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8532) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.53.1...8.53.2) +- Bump CLI from v2.47.0 to v2.50.0 ([#4993](https://github.com/getsentry/sentry-react-native/pull/4993), [#4999](https://github.com/getsentry/sentry-react-native/pull/4999)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2500) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.47.0...2.50.0) +- Bump Bundler Plugins from v3.5.0 to v3.6.1 ([#4994](https://github.com/getsentry/sentry-react-native/pull/4994), [#4998](https://github.com/getsentry/sentry-react-native/pull/4998)) + - [changelog](https://github.com/getsentry/sentry-javascript-bundler-plugins/blob/main/CHANGELOG.md#361) + - [diff](https://github.com/getsentry/sentry-javascript-bundler-plugins/compare/3.5.0...3.6.1) + +## 6.17.0 + +### Features + +- Add experimental flag `enableUnhandledCPPExceptionsV2` on iOS ([#4975](https://github.com/getsentry/sentry-react-native/pull/4975)) + + ```js + import * as Sentry from '@sentry/react-native'; + + Sentry.init({ + _experiments: { + enableUnhandledCPPExceptionsV2: true, + }, + }); + ``` + +### Dependencies + +- Bump CLI from v2.46.0 to v2.47.0 ([#4979](https://github.com/getsentry/sentry-react-native/pull/4979)) + - [changelog](https://github.com/getsentry/sentry-cli/blob/master/CHANGELOG.md#2470) + - [diff](https://github.com/getsentry/sentry-cli/compare/2.46.0...2.47.0) +- Bump Android SDK from v7.22.5 to v7.22.6 ([#4985](https://github.com/getsentry/sentry-react-native/pull/4985)) + - [changelog](https://github.com/getsentry/sentry-java/blob/main/CHANGELOG.md#7226) + - [diff](https://github.com/getsentry/sentry-java/compare/7.22.5...7.22.6) + +## 6.16.1 + +### Fixes + +- Fixes Replay Custom Masking issue on Android ([#4957](https://github.com/getsentry/sentry-react-native/pull/4957)) + +### Dependencies + +- Bump Cocoa SDK from v8.52.1 to v8.53.1 ([#4950](https://github.com/getsentry/sentry-react-native/pull/4950)) + - [changelog](https://github.com/getsentry/sentry-cocoa/blob/main/CHANGELOG.md#8531) + - [diff](https://github.com/getsentry/sentry-cocoa/compare/8.52.1...8.53.1) + +## 6.16.0 + +### Features + +- Introducing `@sentry/react-native/playground` ([#4916](https://github.com/getsentry/sentry-react-native/pull/4916)) + + The new `withSentryPlayground` component allows developers to verify + that the SDK is properly configured and reports errors as expected. + + ```jsx + import * as Sentry from '@sentry/react-native'; + import { withSentryPlayground } from '@sentry/react-native/playground'; + + function App() { + return ...; + } + + export default withSentryPlayground( + Sentry.wrap(App) + ); + ``` + +### Fixes + +- Adds support for React Native 0.80 ([#4938](https://github.com/getsentry/sentry-react-native/pull/4938)) +- Report slow and frozen frames as app start span data ([#4865](https://github.com/getsentry/sentry-react-native/pull/4865)) +- User set by `Sentry.setUser` is prefilled in Feedback Widget ([#4901](https://github.com/getsentry/sentry-react-native/pull/4901)) + - User data are considered from all scopes in the following order current, isolation and global. + ## 6.15.1 ### Dependencies diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index ca64d58f7b..548e75806d 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -10,25 +10,21 @@ This repository contains mono repository structure with multiple React Native an - /dev-packages -> dev packages, ts-3.8 test runner, e2e tests components and runner - /performance-tests -> applications used for measuring performance in CI -# Requirements +# Setting up an Environment -- nodejs 18 (with corepack globally installed) -- yarn version specified in `package.json` (at the moment version 3.6) +We use [Volta](https://volta.sh/) to ensure we use consistent versions of node and yarn. -## Building +`sentry-react-native` is a monorepo containing several packages, and we use `lerna` to manage them. To get started, +install all dependencies, and then perform an initial build, so TypeScript can read all of the linked type definitions. -Install dependencies using: - -```sh -yarn +``` +$ yarn +$ yarn build ``` -Once deps are installed, you can build the project: - -```sh -yarn build +With that, the repo is fully set up and you are ready to run all commands. -# Or in watch mode, for development of the SDK core +# Watch mode, for development of the SDK core cd packages/core yarn build:sdk:watch @@ -46,6 +42,14 @@ yarn test:watch ## Running the sample +First, set up the Sentry CLI token. +A recommended approach is to create a file named `.env.sentry-build-plugin` in the root folder of each sample and add: +```sh +SENTRY_AUTH_TOKEN=... +``` + +To obtain the correct token, log in to Sentry.io, then visit: `https://docs.sentry.io/cli/configuration/#to-authenticate-manually` From there, generate a token following the documentation. + Now we can go into the sample project, install and build it: ```sh diff --git a/dev-packages/e2e-tests/cli.mjs b/dev-packages/e2e-tests/cli.mjs index c9281549f3..9d8f958b80 100755 --- a/dev-packages/e2e-tests/cli.mjs +++ b/dev-packages/e2e-tests/cli.mjs @@ -68,6 +68,25 @@ const testApp = `${e2eDir}/${testAppName}`; const appId = platform === 'ios' ? 'org.reactjs.native.example.RnDiffApp' : 'com.rndiffapp'; const sentryAuthToken = env.SENTRY_AUTH_TOKEN; +function runCodegenIfNeeded(rnVersion, platform, appDir) { + const versionNumber = parseFloat(rnVersion.replace(/[^\d.]/g, '')); + const shouldRunCodegen = platform === 'android' && versionNumber >= 0.80; + + if (shouldRunCodegen) { + console.log(`Running codegen for React Native ${rnVersion}...`); + try { + execSync('./gradlew generateCodegenArtifactsFromSchema', { + stdio: 'inherit', + cwd: path.join(appDir, 'android'), + env: env + }); + console.log('Gradle codegen task completed successfully'); + } catch (error) { + console.error('Codegen failed:', error.message); + } + } +} + // Build and publish the SDK - we only need to do this once in CI. // Locally, we may want to get updates from the latest build so do it on every app build. if (actions.includes('create') || (env.CI === undefined && actions.includes('build'))) { @@ -198,6 +217,8 @@ if (actions.includes('build')) { appProduct = `${appDir}/ios/DerivedData/Build/Products/${buildType}-iphonesimulator/${appName}.app`; } else if (platform == 'android') { + runCodegenIfNeeded(RNVersion, platform, appDir); + execSync(`./gradlew assemble${buildType} -PreactNativeArchitectures=x86 --no-daemon`, { stdio: 'inherit', cwd: `${appDir}/android`, diff --git a/dev-packages/e2e-tests/maestro/feedback.yml b/dev-packages/e2e-tests/maestro/feedback.yml index ce7d79b89d..92f17871c4 100644 --- a/dev-packages/e2e-tests/maestro/feedback.yml +++ b/dev-packages/e2e-tests/maestro/feedback.yml @@ -17,3 +17,13 @@ jsEngine: graaljs file: feedback/happyFlow-android.yml when: platform: Android + +- runFlow: + file: feedback/captureFlow-ios.yml + when: + platform: iOS + +- runFlow: + file: feedback/captureFlow-android.yml + when: + platform: Android diff --git a/dev-packages/e2e-tests/maestro/feedback/captureFlow-android.yml b/dev-packages/e2e-tests/maestro/feedback/captureFlow-android.yml new file mode 100644 index 0000000000..a949b7a89b --- /dev/null +++ b/dev-packages/e2e-tests/maestro/feedback/captureFlow-android.yml @@ -0,0 +1,54 @@ +# This is a happy path test for the feedback widget on Android. +# It verifies that the feedback form can be opened, filled out, and submitted successfully +appId: ${APP_ID} +jsEngine: graaljs +--- + +# Show feedback button +- tapOn: 'Feedback' + +# Open feedback widget +- tapOn: 'Report a Bug' + +# Assert that the feedback form is visible +- extendedWaitUntil: + visible: 'Report a Bug' + timeout: 5_000 + +# Fill out name field +- tapOn: 'Your Name' +- inputText: 'John Doe' + +# Fill out email field +- tapOn: 'your.email@example.org' +- inputText: 'test@email.com' + +# Fill out message field +- tapOn: "What's the bug? What did you expect?" +- inputText: 'This is a test feedback message with a screenshot from CI e2e tests' + +# Take screenshot +- scrollUntilVisible: + element: + text: 'Take a screenshot' +- tapOn: 'Take a screenshot' +- tapOn: 'Take Screenshot' + +# Assert that the feedback form is visible +- extendedWaitUntil: + visible: 'Report a Bug' + timeout: 5_000 + +# Hide keyboard by tapping on a non-tappable element +- tapOn: 'Email' + +# Submit feedback +- scrollUntilVisible: + element: + text: 'Send Bug Report' +- tapOn: 'Send Bug Report' +- assertVisible: 'Thank you for your report!' +- tapOn: 'OK' + +# Verify feedback form is closed and the home screen is visible +- assertVisible: 'Welcome to React Native' diff --git a/dev-packages/e2e-tests/maestro/feedback/captureFlow-ios.yml b/dev-packages/e2e-tests/maestro/feedback/captureFlow-ios.yml new file mode 100644 index 0000000000..6a3ebe8023 --- /dev/null +++ b/dev-packages/e2e-tests/maestro/feedback/captureFlow-ios.yml @@ -0,0 +1,58 @@ +# This is a happy path test for the feedback widget on iOS. +# It verifies that the feedback form can be opened, filled out, and submitted successfully +appId: ${APP_ID} +jsEngine: graaljs +--- + +# Show feedback button +- tapOn: 'Feedback' + +# Open feedback widget +- tapOn: + id: 'sentry-feedback-button' + +# Assert that the feedback form is visible +- extendedWaitUntil: + visible: + id: 'sentry-feedback-form-title' + timeout: 5_000 + +# Fill out name field +- tapOn: + id: 'sentry-feedback-name-input' +- inputText: 'John Doe' + +# Fill out email field +- tapOn: + id: 'sentry-feedback-email-input' +- inputText: 'test@email.com' + +# Fill out message field +- tapOn: + id: 'sentry-feedback-message-input' +- inputText: 'This is a test feedback message with a screenshot from CI e2e tests' + +# Take screenshot +- scrollUntilVisible: + element: + id: 'sentry-feedback-take-screenshot-button' +- tapOn: + id: 'sentry-feedback-take-screenshot-button' +- tapOn: + id: 'sentry-feedback-screenshot-button' + +# Hide keyboard by tapping on a non-tappable element +- tapOn: + id: 'sentry-logo' + +# Submit feedback +- scrollUntilVisible: + element: + id: 'sentry-feedback-submit-button' +- tapOn: + id: 'sentry-feedback-submit-button' +- assertVisible: 'Thank you for your report!' +- tapOn: 'OK' + +# Verify feedback form is closed and the home screen is visible +- assertVisible: 'Welcome to React Native' diff --git a/dev-packages/e2e-tests/maestro/feedback/happyFlow-android.yml b/dev-packages/e2e-tests/maestro/feedback/happyFlow-android.yml index 221b0cbf84..cc81ab8dd4 100644 --- a/dev-packages/e2e-tests/maestro/feedback/happyFlow-android.yml +++ b/dev-packages/e2e-tests/maestro/feedback/happyFlow-android.yml @@ -27,6 +27,9 @@ jsEngine: graaljs - tapOn: "What's the bug? What did you expect?" - inputText: 'This is a test feedback message from CI e2e tests' +# Hide keyboard by tapping on a non-tappable element +- tapOn: 'Email' + # Submit feedback - scrollUntilVisible: element: diff --git a/dev-packages/e2e-tests/maestro/feedback/happyFlow-ios.yml b/dev-packages/e2e-tests/maestro/feedback/happyFlow-ios.yml index 7f8c3340b1..41d36cf15c 100644 --- a/dev-packages/e2e-tests/maestro/feedback/happyFlow-ios.yml +++ b/dev-packages/e2e-tests/maestro/feedback/happyFlow-ios.yml @@ -32,6 +32,10 @@ jsEngine: graaljs id: 'sentry-feedback-message-input' - inputText: 'This is a test feedback message from CI e2e tests' +# Hide keyboard by tapping on a non-tappable element +- tapOn: + id: 'sentry-logo' + # Submit feedback - scrollUntilVisible: element: diff --git a/dev-packages/e2e-tests/package.json b/dev-packages/e2e-tests/package.json index 3c4fc4046e..90b2021a47 100644 --- a/dev-packages/e2e-tests/package.json +++ b/dev-packages/e2e-tests/package.json @@ -1,6 +1,6 @@ { "name": "sentry-react-native-e2e-tests", - "version": "6.15.1", + "version": "7.4.0", "private": true, "description": "Sentry React Native End to End Tests Library", "main": "dist/index.js", @@ -13,13 +13,13 @@ "devDependencies": { "@babel/preset-env": "^7.25.3", "@babel/preset-typescript": "^7.18.6", - "@sentry/core": "8.54.0", - "@sentry/react-native": "6.15.1", + "@sentry/core": "10.20.0", + "@sentry/react-native": "7.4.0", "@types/node": "^20.9.3", "@types/react": "^18.2.64", "appium": "2.4.1", - "appium-uiautomator2-driver": "2.39.0", - "appium-xcuitest-driver": "5.13.0", + "appium-uiautomator2-driver": "2.43.4", + "appium-xcuitest-driver": "5.15.1", "babel-jest": "^29.7.0", "jest": "^29.7.0", "react": "18.3.1", diff --git a/dev-packages/e2e-tests/patch-scripts/rn.patch.app.build.gradle.js b/dev-packages/e2e-tests/patch-scripts/rn.patch.app.build.gradle.js index 8eec66240b..c6ae9712af 100755 --- a/dev-packages/e2e-tests/patch-scripts/rn.patch.app.build.gradle.js +++ b/dev-packages/e2e-tests/patch-scripts/rn.patch.app.build.gradle.js @@ -4,15 +4,15 @@ const fs = require('fs'); const { argv } = require('process'); const parseArgs = require('minimist'); -const { logger } = require('@sentry/core'); -logger.enable(); +const { debug } = require('@sentry/core'); +debug.enable(); const args = parseArgs(argv.slice(2)); if (!args['app-build-gradle']) { throw new Error('Missing --app-build-gradle'); } -logger.info('Patching app/build.gradle', args['app-build-gradle']); +debug.log('Patching app/build.gradle', args['app-build-gradle']); const sentryGradlePatch = ` apply from: new File(["node", "--print", "require.resolve('@sentry/react-native/package.json')"].execute().text.trim(), "../sentry.gradle") @@ -26,7 +26,7 @@ if (!isPatched) { const patched = buildGradle.replace(reactNativeGradleRex, m => sentryGradlePatch + m); fs.writeFileSync(args['app-build-gradle'], patched); - logger.info('Patched app/build.gradle successfully!'); + debug.log('Patched app/build.gradle successfully!'); } else { - logger.info('app/build.gradle is already patched!'); + debug.log('app/build.gradle is already patched!'); } diff --git a/dev-packages/e2e-tests/patch-scripts/rn.patch.app.js b/dev-packages/e2e-tests/patch-scripts/rn.patch.app.js index 48660f1a56..2a6ac3b14d 100755 --- a/dev-packages/e2e-tests/patch-scripts/rn.patch.app.js +++ b/dev-packages/e2e-tests/patch-scripts/rn.patch.app.js @@ -5,8 +5,8 @@ const path = require('path'); const { argv, env } = require('process'); const parseArgs = require('minimist'); -const { logger } = require('@sentry/core'); -logger.enable(); +const { debug } = require('@sentry/core'); +debug.enable(); const SENTRY_RELEASE = env.SENTRY_RELEASE; const SENTRY_DIST = env.SENTRY_DIST; @@ -16,7 +16,7 @@ if (!args.app) { throw new Error('Missing --app'); } -logger.info('Patching RN App.(js|tsx)', args.app); +debug.log('Patching RN App.(js|tsx)', args.app); const initPatch = ` import * as Sentry from '@sentry/react-native'; @@ -32,13 +32,17 @@ Sentry.init({ }, integrations: [ Sentry.mobileReplayIntegration(), + Sentry.feedbackIntegration({ + enableTakeScreenshot: true, + }), ], }); `; const e2eComponentPatch = ''; const lastImportRex = /^([^]*)(import\s+[^;]*?;$)/m; const patchRex = '@sentry/react-native'; -const headerComponentRex = / true,'); @@ -32,11 +32,11 @@ if (shouldPatch) { enableHermes ? ':hermes_enabled => true,' : ':hermes_enabled => false,', ); if (enableHermes) { - logger.info('Patching Podfile for Hermes'); + debug.log('Patching Podfile for Hermes'); } else { - logger.info('Patching Podfile for JSC'); + debug.log('Patching Podfile for JSC'); } fs.writeFileSync(args['pod-file'], patched); } else { - logger.info('Podfile is already patched!'); + debug.log('Podfile is already patched!'); } diff --git a/dev-packages/e2e-tests/patch-scripts/rn.patch.xcode.js b/dev-packages/e2e-tests/patch-scripts/rn.patch.xcode.js index d044817df2..072a885720 100755 --- a/dev-packages/e2e-tests/patch-scripts/rn.patch.xcode.js +++ b/dev-packages/e2e-tests/patch-scripts/rn.patch.xcode.js @@ -6,8 +6,8 @@ const { argv } = require('process'); const xcode = require('xcode'); const parseArgs = require('minimist'); const semver = require('semver'); -const { logger } = require('@sentry/core'); -logger.enable(); +const { debug } = require('@sentry/core'); +debug.enable(); const args = parseArgs(argv.slice(2)); if (!args.project) { @@ -17,7 +17,7 @@ if (!args['rn-version']) { throw new Error('Missing --rn-version'); } -logger.info('Patching Xcode project', args.project, 'for RN version', args['rn-version']); +debug.log('Patching Xcode project', args.project, 'for RN version', args['rn-version']); const newBundleScriptRNVersion = '0.69.0-rc.0'; @@ -29,7 +29,7 @@ const symbolsScript = ` `; const symbolsPatchRegex = /sentry-cli\s+(upload-dsym|debug-files upload)/; if (semver.satisfies(args['rn-version'], `< ${newBundleScriptRNVersion}`, { includePrerelease: true })) { - logger.info('Applying old bundle script patch'); + debug.log('Applying old bundle script patch'); bundleScript = ` export NODE_BINARY=node ../node_modules/@sentry/react-native/scripts/sentry-xcode.sh ../node_modules/react-native/scripts/react-native-xcode.sh @@ -37,7 +37,7 @@ export NODE_BINARY=node bundleScriptRegex = /(packager|scripts)\/react-native-xcode\.sh\b/; bundlePatchRegex = /sentry-cli\s+react-native[\s-]xcode/; } else if (semver.satisfies(args['rn-version'], `>= ${newBundleScriptRNVersion}`, { includePrerelease: true })) { - logger.info('Applying new bundle script patch'); + debug.log('Applying new bundle script patch'); bundleScript = ` WITH_ENVIRONMENT="../node_modules/react-native/scripts/xcode/with-environment.sh" REACT_NATIVE_XCODE="../node_modules/react-native/scripts/react-native-xcode.sh" @@ -62,38 +62,32 @@ for (const key in buildPhasesRaw) { } } -buildPhases.forEach((phase) => { +buildPhases.forEach(phase => { const isBundleReactNative = phase.shellScript.match(bundleScriptRegex); const isPatched = phase.shellScript.match(bundlePatchRegex); if (!isBundleReactNative) { return; } if (isPatched) { - logger.warn('Xcode project Bundle RN Build phase already patched'); + debug.warn('Xcode project Bundle RN Build phase already patched'); return; } phase.shellScript = JSON.stringify(bundleScript); - logger.info('Patched Xcode project Bundle RN Build phase'); + debug.log('Patched Xcode project Bundle RN Build phase'); }); -const isSymbolsPhase = (phase) => phase.shellScript.match(symbolsPatchRegex); +const isSymbolsPhase = phase => phase.shellScript.match(symbolsPatchRegex); const areSymbolsPatched = buildPhases.some(isSymbolsPhase); if (!areSymbolsPatched) { - project.addBuildPhase( - [], - 'PBXShellScriptBuildPhase', - 'Upload Debug Symbols to Sentry', - null, - { - shellPath: '/bin/sh', - shellScript: symbolsScript, - }, - ); - logger.info('Added Xcode project Upload Debug Symbols Build phase'); + project.addBuildPhase([], 'PBXShellScriptBuildPhase', 'Upload Debug Symbols to Sentry', null, { + shellPath: '/bin/sh', + shellScript: symbolsScript, + }); + debug.log('Added Xcode project Upload Debug Symbols Build phase'); } else { - logger.warn('Xcode project Upload Debug Symbols Build phase already patched'); + debug.warn('Xcode project Upload Debug Symbols Build phase already patched'); } fs.writeFileSync(args.project, project.writeSync()); -logger.info('Patched Xcode project successfully!'); +debug.log('Patched Xcode project successfully!'); diff --git a/dev-packages/type-check/package.json b/dev-packages/type-check/package.json index ca2c6623d7..a9b55d9afc 100644 --- a/dev-packages/type-check/package.json +++ b/dev-packages/type-check/package.json @@ -1,7 +1,7 @@ { "name": "sentry-react-native-type-check", "private": true, - "version": "6.15.1", + "version": "7.4.0", "scripts": { "type-check": "./run-type-check.sh" } diff --git a/dev-packages/type-check/ts3.8-test/index.ts b/dev-packages/type-check/ts3.8-test/index.ts index 1e9fda3cd2..d6cc248482 100644 --- a/dev-packages/type-check/ts3.8-test/index.ts +++ b/dev-packages/type-check/ts3.8-test/index.ts @@ -3,6 +3,8 @@ declare global { interface IDBObjectStore {} interface Window { fetch: any; + setTimeout: any; + document: any; } interface ShadowRoot {} interface BufferSource {} @@ -19,6 +21,8 @@ declare global { redirectCount: number; } interface PerformanceEntry {} + interface Performance {} + interface PerformanceNavigationTiming {} } declare module 'react-native' { diff --git a/dev-packages/type-check/ts3.8-test/tsconfig.build.json b/dev-packages/type-check/ts3.8-test/tsconfig.build.json index 28d4363cb9..5b4837604c 100644 --- a/dev-packages/type-check/ts3.8-test/tsconfig.build.json +++ b/dev-packages/type-check/ts3.8-test/tsconfig.build.json @@ -3,7 +3,7 @@ "index.ts", ], "compilerOptions": { - "skipLibCheck": false, + "skipLibCheck": true, "noEmit": true, "importHelpers": true, "types": [], diff --git a/dev-packages/utils/package.json b/dev-packages/utils/package.json index 87c27ab3ec..5cbe826f50 100644 --- a/dev-packages/utils/package.json +++ b/dev-packages/utils/package.json @@ -1,6 +1,6 @@ { "name": "sentry-react-native-samples-utils", - "version": "6.15.1", + "version": "7.4.0", "description": "Internal Samples Utils", "main": "index.js", "license": "MIT", diff --git a/lerna.json b/lerna.json index 28dd6410f8..aeb4b37f9e 100644 --- a/lerna.json +++ b/lerna.json @@ -1,6 +1,6 @@ { "$schema": "node_modules/lerna/schemas/lerna-schema.json", - "version": "6.15.1", + "version": "7.4.0", "packages": [ "packages/*", "dev-packages/*", diff --git a/package.json b/package.json index 7af7287f13..7597a4acac 100644 --- a/package.json +++ b/package.json @@ -7,13 +7,13 @@ "clean": "lerna run clean", "circularDepCheck": "lerna run circularDepCheck", "test": "lerna run test", - "fix": "run-s fix:lerna fix:android fix:clang fix:swift fix:kotlin", + "fix": "run-s fix:lerna fix:android fix:kotlin fix:clang fix:swift", "fix:lerna": "lerna run fix", "fix:android": "run-s 'java:format fix' java:pmd", "fix:clang": "run-s 'clang:format fix'", "fix:swift": "run-s 'swift:lint fix'", "fix:kotlin": "npx ktlint --relative --format '!**/node_modules/**'", - "lint": "run-s lint:lerna lint:android lint:clang lint:swift lint:kotlin", + "lint": "run-s lint:lerna lint:android lint:kotlin lint:clang lint:swift ", "lint:lerna": "lerna run lint", "lint:android": "run-s 'java:format lint' java:pmd", "lint:clang": "run-s 'clang:format lint'", @@ -28,10 +28,8 @@ "set-version-samples": "lerna run set-version" }, "devDependencies": { - "@expo/swiftlint": "^0.57.1", "@naturalcycles/ktlint": "^1.13.0", - "@sentry/cli": "2.46.0", - "clang-format": "^1.8.0", + "@sentry/cli": "2.56.1", "downlevel-dts": "^0.11.0", "google-java-format": "^1.4.0", "lerna": "^8.1.8", @@ -56,8 +54,13 @@ "Appium has a dependency on @xmldom/xmldom@^0.x, which causes chromedrive build to fail yarn install", "See: https://github.com/appium/appium-chromedriver/pull/424" ], + "volta": { + "node": "18.20.8", + "yarn": "3.6.4" + }, "resolutions": { - "appium-chromedriver@npm:5.6.73/@xmldom/xmldom": "0.8.10" + "appium-chromedriver@npm:5.6.73/@xmldom/xmldom": "0.8.10", + "form-data": "4.0.4" }, "version": "0.0.0", "name": "sentry-react-native", diff --git a/packages/core/.eslintrc.js b/packages/core/.eslintrc.js index 8916a1cfd9..3ad06277ab 100644 --- a/packages/core/.eslintrc.js +++ b/packages/core/.eslintrc.js @@ -19,6 +19,8 @@ module.exports = { 'metro.d.ts', 'plugin/build/**/*', 'expo.d.ts', + 'playground.js', + 'playground.d.ts', ], overrides: [ { @@ -38,6 +40,7 @@ module.exports = { '@typescript-eslint/no-empty-function': 'off', '@typescript-eslint/no-explicit-any': 'off', '@typescript-eslint/unbound-method': 'off', + 'import/first': 'off', }, }, { diff --git a/packages/core/RNSentry.podspec b/packages/core/RNSentry.podspec index d24d88d420..be8c202bf5 100644 --- a/packages/core/RNSentry.podspec +++ b/packages/core/RNSentry.podspec @@ -6,15 +6,24 @@ rn_package = parse_rn_package_json() rn_version = get_rn_version(rn_package) is_hermes_default = is_hermes_default(rn_version) is_profiling_supported = is_profiling_supported(rn_version) +is_new_hermes_runtime = is_new_hermes_runtime(rn_version) -folly_flags = ' -DFOLLY_NO_CONFIG -DFOLLY_MOBILE=1 -DFOLLY_USE_LIBCPP=1' -folly_compiler_flags = folly_flags + ' ' + '-Wno-comma -Wno-shorten-64-to-32' +# Use different Folly configuration for RN 0.80.0+ +if should_use_folly_flags(rn_version) + # For older RN versions, keep the original Folly configuration + folly_flags = ' -DFOLLY_NO_CONFIG -DFOLLY_MOBILE=1 -DFOLLY_USE_LIBCPP=1' + folly_compiler_flags = folly_flags + ' ' + '-Wno-comma -Wno-shorten-64-to-32' +else + # For RN 0.80+, don't use the incompatible Folly flags + folly_compiler_flags = '' +end is_new_arch_enabled = ENV["RCT_NEW_ARCH_ENABLED"] == "1" is_using_hermes = (ENV['USE_HERMES'] == nil && is_hermes_default) || ENV['USE_HERMES'] == '1' new_arch_enabled_flag = (is_new_arch_enabled ? folly_compiler_flags + " -DRCT_NEW_ARCH_ENABLED" : "") sentry_profiling_supported_flag = (is_profiling_supported ? " -DSENTRY_PROFILING_SUPPORTED=1" : "") -other_cflags = "$(inherited)" + new_arch_enabled_flag + sentry_profiling_supported_flag +new_hermes_runtime_flag = (is_new_hermes_runtime ? " -DNEW_HERMES_RUNTIME" : "") +other_cflags = "$(inherited)" + new_arch_enabled_flag + sentry_profiling_supported_flag + new_hermes_runtime_flag Pod::Spec.new do |s| s.name = 'RNSentry' @@ -37,7 +46,7 @@ Pod::Spec.new do |s| s.compiler_flags = other_cflags - s.dependency 'Sentry/HybridSDK', '8.52.1' + s.dependency 'Sentry/HybridSDK', '8.57.0' if defined? install_modules_dependencies # Default React Native dependencies for 0.71 and above (new and legacy architecture) diff --git a/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/RNSentryStartTest.kt b/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/RNSentryStartTest.kt index fa177159e5..5ec84c6532 100644 --- a/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/RNSentryStartTest.kt +++ b/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/RNSentryStartTest.kt @@ -248,4 +248,91 @@ class RNSentryStartTest { assertEquals("android", result?.getTag("event.origin")) assertEquals("java", result?.getTag("event.environment")) } + + @Test + fun `trySetIgnoreErrors sets only regex patterns`() { + val options = SentryAndroidOptions() + val rnOptions = + JavaOnlyMap.of( + "ignoreErrorsRegex", + com.facebook.react.bridge.JavaOnlyArray + .of("^Foo.*", "Bar$"), + ) + module.trySetIgnoreErrors(options, rnOptions) + assertEquals(listOf("^Foo.*", "Bar$"), options.ignoredErrors!!.map { it.filterString }) + } + + @Test + fun `trySetIgnoreErrors sets only string patterns`() { + val options = SentryAndroidOptions() + val rnOptions = + JavaOnlyMap.of( + "ignoreErrorsStr", + com.facebook.react.bridge.JavaOnlyArray + .of("ExactError", "AnotherError"), + ) + module.trySetIgnoreErrors(options, rnOptions) + assertEquals(listOf(".*\\QExactError\\E.*", ".*\\QAnotherError\\E.*"), options.ignoredErrors!!.map { it.filterString }) + } + + @Test + fun `trySetIgnoreErrors sets both regex and string patterns`() { + val options = SentryAndroidOptions() + val rnOptions = + JavaOnlyMap.of( + "ignoreErrorsRegex", + com.facebook.react.bridge.JavaOnlyArray + .of("^Foo.*"), + "ignoreErrorsStr", + com.facebook.react.bridge.JavaOnlyArray + .of("ExactError"), + ) + module.trySetIgnoreErrors(options, rnOptions) + assertEquals(listOf("^Foo.*", ".*\\QExactError\\E.*"), options.ignoredErrors!!.map { it.filterString }) + } + + @Test + fun `trySetIgnoreErrors sets nothing if neither is present`() { + val options = SentryAndroidOptions() + val rnOptions = JavaOnlyMap.of() + module.trySetIgnoreErrors(options, rnOptions) + assertNull(options.ignoredErrors) + } + + @Test + fun `trySetIgnoreErrors with string containing regex special characters should match literally if Pattern_quote is used`() { + val options = SentryAndroidOptions() + val special = "I like chocolate (and tomato)." + val rnOptions = + JavaOnlyMap.of( + "ignoreErrorsStr", + com.facebook.react.bridge.JavaOnlyArray + .of(special), + ) + module.trySetIgnoreErrors(options, rnOptions) + + assertEquals(listOf(".*\\QI like chocolate (and tomato).\\E.*"), options.ignoredErrors!!.map { it.filterString }) + + val regex = Regex(options.ignoredErrors!![0].filterString) + assertTrue(regex.matches("I like chocolate (and tomato).")) + assertTrue(regex.matches(" I like chocolate (and tomato). ")) + assertTrue(regex.matches("I like chocolate (and tomato). And vanilla.")) + } + + @Test + fun `trySetIgnoreErrors with string containing star should not match everything if Pattern_quote is used`() { + val options = SentryAndroidOptions() + val special = "Error*WithStar" + val rnOptions = + JavaOnlyMap.of( + "ignoreErrorsStr", + com.facebook.react.bridge.JavaOnlyArray + .of(special), + ) + module.trySetIgnoreErrors(options, rnOptions) + assertEquals(listOf(".*\\QError*WithStar\\E.*"), options.ignoredErrors!!.map { it.filterString }) + + val regex = Regex(options.ignoredErrors!![0].filterString) + assertTrue(regex.matches("Error*WithStar")) + } } diff --git a/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/replay/RNSentryReplayMaskManagerTest.kt b/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/replay/RNSentryReplayMaskManagerTest.kt new file mode 100644 index 0000000000..33e4e1aaa7 --- /dev/null +++ b/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/replay/RNSentryReplayMaskManagerTest.kt @@ -0,0 +1,39 @@ +package io.sentry.react.replay + +import com.facebook.react.module.annotations.ReactModule +import org.junit.Assert.assertEquals +import org.junit.Assert.assertNotNull +import org.junit.Before +import org.junit.Test +import org.junit.runner.RunWith +import org.junit.runners.JUnit4 +import org.mockito.MockitoAnnotations + +@RunWith(JUnit4::class) +class RNSentryReplayMaskManagerTest { + private val expectedName = RNSentryReplayMaskManagerImpl.REACT_CLASS + + private lateinit var manager: RNSentryReplayMaskManager + + @Before + fun setUp() { + MockitoAnnotations.openMocks(this) + manager = RNSentryReplayMaskManager() + } + + @Test + fun `getName returns correct react class name`() { + assertEquals(expectedName, manager.getName()) + } + + @Test + fun `module annotation name matches getName result`() { + val annotation = manager.javaClass.getAnnotation(ReactModule::class.java) + assertNotNull("ReactModule annotation should be present", annotation) + assertEquals( + "Annotation name should match getName() result", + expectedName, + annotation?.name, + ) + } +} diff --git a/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/replay/RNSentryReplayUnmaskManagerTest.kt b/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/replay/RNSentryReplayUnmaskManagerTest.kt new file mode 100644 index 0000000000..eb447435fd --- /dev/null +++ b/packages/core/RNSentryAndroidTester/app/src/test/java/io/sentry/react/replay/RNSentryReplayUnmaskManagerTest.kt @@ -0,0 +1,39 @@ +package io.sentry.react.replay + +import com.facebook.react.module.annotations.ReactModule +import org.junit.Assert.assertEquals +import org.junit.Assert.assertNotNull +import org.junit.Before +import org.junit.Test +import org.junit.runner.RunWith +import org.junit.runners.JUnit4 +import org.mockito.MockitoAnnotations + +@RunWith(JUnit4::class) +class RNSentryReplayUnmaskManagerTest { + private val expectedName = RNSentryReplayUnmaskManagerImpl.REACT_CLASS + + private lateinit var manager: RNSentryReplayUnmaskManager + + @Before + fun setUp() { + MockitoAnnotations.openMocks(this) + manager = RNSentryReplayUnmaskManager() + } + + @Test + fun `getName returns correct react class name`() { + assertEquals(expectedName, manager.getName()) + } + + @Test + fun `module annotation name matches getName result`() { + val annotation = manager.javaClass.getAnnotation(ReactModule::class.java) + assertNotNull("ReactModule annotation should be present", annotation) + assertEquals( + "Annotation name should match getName() result", + expectedName, + annotation?.name, + ) + } +} diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/project.pbxproj b/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/project.pbxproj index c9297fbfd1..8e389c0b4c 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/project.pbxproj +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/project.pbxproj @@ -8,7 +8,7 @@ /* Begin PBXBuildFile section */ 332D33472CDBDBB600547D76 /* RNSentryReplayOptionsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 332D33462CDBDBB600547D76 /* RNSentryReplayOptionsTests.swift */; }; - 3339C4812D6625570088EB3A /* RNSentryUserTests.mm in Sources */ = {isa = PBXBuildFile; fileRef = 3339C4802D6625570088EB3A /* RNSentryUserTests.mm */; }; + 3339C4812D6625570088EB3A /* RNSentryUserTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 3339C4802D6625570088EB3A /* RNSentryUserTests.m */; }; 336084392C32E382008CC412 /* RNSentryReplayBreadcrumbConverterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 336084382C32E382008CC412 /* RNSentryReplayBreadcrumbConverterTests.swift */; }; 3380C6C42CE25ECA0018B9B6 /* RNSentryReplayPostInitTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3380C6C32CE25ECA0018B9B6 /* RNSentryReplayPostInitTests.swift */; }; 33958C692BFCF12600AD1FB6 /* RNSentryOnDrawReporterTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 33958C682BFCF12600AD1FB6 /* RNSentryOnDrawReporterTests.m */; }; @@ -22,7 +22,7 @@ 33DEDFEA2D8DBE67006066E4 /* RNSentryOnDrawReporterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33DEDFE92D8DBE5B006066E4 /* RNSentryOnDrawReporterTests.swift */; }; 33DEDFED2D8DC825006066E4 /* RNSentryOnDrawReporter+Test.mm in Sources */ = {isa = PBXBuildFile; fileRef = 33DEDFEC2D8DC820006066E4 /* RNSentryOnDrawReporter+Test.mm */; }; 33DEDFF02D9185EB006066E4 /* RNSentryTimeToDisplayTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33DEDFEF2D9185E3006066E4 /* RNSentryTimeToDisplayTests.swift */; }; - 33F58AD02977037D008F60EA /* RNSentryTests.mm in Sources */ = {isa = PBXBuildFile; fileRef = 33F58ACF2977037D008F60EA /* RNSentryTests.mm */; }; + 33F58AD02977037D008F60EA /* RNSentryTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 33F58ACF2977037D008F60EA /* RNSentryTests.m */; }; AEFB00422CC90C4B00EC8A9A /* RNSentryBreadcrumbTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3360843C2C340C76008CC412 /* RNSentryBreadcrumbTests.swift */; }; B5859A50A3E865EF5E61465A /* libPods-RNSentryCocoaTesterTests.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 650CB718ACFBD05609BF2126 /* libPods-RNSentryCocoaTesterTests.a */; }; /* End PBXBuildFile section */ @@ -35,10 +35,10 @@ 332D33492CDCC8E100547D76 /* RNSentryTests.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = RNSentryTests.h; sourceTree = ""; }; 332D334A2CDCC8EB00547D76 /* RNSentryCocoaTesterTests-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "RNSentryCocoaTesterTests-Bridging-Header.h"; sourceTree = ""; }; 3339C47F2D6625260088EB3A /* RNSentry+Test.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "RNSentry+Test.h"; sourceTree = ""; }; - 3339C4802D6625570088EB3A /* RNSentryUserTests.mm */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.objcpp; path = RNSentryUserTests.mm; sourceTree = ""; }; 333B58A82D35BA93000F8D04 /* RNSentryStart.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = RNSentryStart.h; path = ../ios/RNSentryStart.h; sourceTree = SOURCE_ROOT; }; 333B58A92D35BB2D000F8D04 /* RNSentryStart+Test.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = "RNSentryStart+Test.h"; path = "RNSentryCocoaTesterTests/RNSentryStart+Test.h"; sourceTree = SOURCE_ROOT; }; 333B58AF2D36A7FD000F8D04 /* RNSentrySDK.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = RNSentrySDK.h; path = ../ios/RNSentrySDK.h; sourceTree = SOURCE_ROOT; }; + 3339C4802D6625570088EB3A /* RNSentryUserTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = RNSentryUserTests.m; sourceTree = ""; }; 336084382C32E382008CC412 /* RNSentryReplayBreadcrumbConverterTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = RNSentryReplayBreadcrumbConverterTests.swift; sourceTree = ""; }; 3360843A2C32E3A8008CC412 /* RNSentryReplayBreadcrumbConverter.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = RNSentryReplayBreadcrumbConverter.h; path = ../ios/RNSentryReplayBreadcrumbConverter.h; sourceTree = ""; }; 3360843C2C340C76008CC412 /* RNSentryBreadcrumbTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RNSentryBreadcrumbTests.swift; sourceTree = ""; }; @@ -66,7 +66,7 @@ 33DEDFEC2D8DC820006066E4 /* RNSentryOnDrawReporter+Test.mm */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.objcpp; path = "RNSentryOnDrawReporter+Test.mm"; sourceTree = ""; }; 33DEDFEE2D8DD431006066E4 /* RNSentryTimeToDisplay.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = RNSentryTimeToDisplay.h; path = ../ios/RNSentryTimeToDisplay.h; sourceTree = SOURCE_ROOT; }; 33DEDFEF2D9185E3006066E4 /* RNSentryTimeToDisplayTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RNSentryTimeToDisplayTests.swift; sourceTree = ""; }; - 33F58ACF2977037D008F60EA /* RNSentryTests.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = RNSentryTests.mm; sourceTree = ""; }; + 33F58ACF2977037D008F60EA /* RNSentryTests.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = RNSentryTests.m; sourceTree = ""; }; 650CB718ACFBD05609BF2126 /* libPods-RNSentryCocoaTesterTests.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-RNSentryCocoaTesterTests.a"; sourceTree = BUILT_PRODUCTS_DIR; }; E2321E7CFA55AB617247098E /* Pods-RNSentryCocoaTesterTests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RNSentryCocoaTesterTests.debug.xcconfig"; path = "Target Support Files/Pods-RNSentryCocoaTesterTests/Pods-RNSentryCocoaTesterTests.debug.xcconfig"; sourceTree = ""; }; /* End PBXFileReference section */ @@ -125,8 +125,8 @@ 332D334A2CDCC8EB00547D76 /* RNSentryCocoaTesterTests-Bridging-Header.h */, 332D33492CDCC8E100547D76 /* RNSentryTests.h */, 336084382C32E382008CC412 /* RNSentryReplayBreadcrumbConverterTests.swift */, - 33F58ACF2977037D008F60EA /* RNSentryTests.mm */, - 3339C4802D6625570088EB3A /* RNSentryUserTests.mm */, + 33F58ACF2977037D008F60EA /* RNSentryTests.m */, + 3339C4802D6625570088EB3A /* RNSentryUserTests.m */, 33AFDFEC2B8D14B300AAB120 /* RNSentryFramesTrackerListenerTests.m */, 33AFDFEE2B8D14C200AAB120 /* RNSentryFramesTrackerListenerTests.h */, 33AFDFF02B8D15E500AAB120 /* RNSentryDependencyContainerTests.m */, @@ -309,8 +309,8 @@ 339C6C422D3FD3AE00CA72ED /* RNSentryStartFromFileTests.swift in Sources */, 336084392C32E382008CC412 /* RNSentryReplayBreadcrumbConverterTests.swift in Sources */, 33DEDFED2D8DC825006066E4 /* RNSentryOnDrawReporter+Test.mm in Sources */, - 33F58AD02977037D008F60EA /* RNSentryTests.mm in Sources */, - 3339C4812D6625570088EB3A /* RNSentryUserTests.mm in Sources */, + 33F58AD02977037D008F60EA /* RNSentryTests.m in Sources */, + 3339C4812D6625570088EB3A /* RNSentryUserTests.m in Sources */, 33DEDFF02D9185EB006066E4 /* RNSentryTimeToDisplayTests.swift in Sources */, 3380C6C42CE25ECA0018B9B6 /* RNSentryReplayPostInitTests.swift in Sources */, 33AFDFED2B8D14B300AAB120 /* RNSentryFramesTrackerListenerTests.m in Sources */, diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/xcshareddata/xcschemes/RNSentryCocoaTester.xcscheme b/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/xcshareddata/xcschemes/RNSentryCocoaTester.xcscheme index b671b3f89e..caf19dfc2d 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/xcshareddata/xcschemes/RNSentryCocoaTester.xcscheme +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTester.xcodeproj/xcshareddata/xcschemes/RNSentryCocoaTester.xcscheme @@ -30,7 +30,7 @@ + parallelizable = "NO"> -@interface -RNSentry (RNSentryInternal) +@interface RNSentry (RNSentryInternal) + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys otherUserKeys:(NSDictionary *)userDataKeys; diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.h b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.h index 3b3055e2f3..c987776703 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.h +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.h @@ -2,11 +2,6 @@ #import #import -@interface -SentrySDK (PrivateTests) -- (nullable SentryOptions *)options; -@end - @interface SentryDependencyContainer : NSObject + (instancetype)sharedInstance; @property (nonatomic, strong) SentryFramesTracker *framesTracker; diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.m b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.m index ab81eb658b..1cef19682c 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.m +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryDependencyContainerTests.m @@ -22,7 +22,7 @@ - (void)testRNSentryDependencyContainerInitializesFrameTracker OCMStub([(SentryDependencyContainer *)sentryDependencyContainerMock framesTracker]) .andReturn(frameTrackerMock); - RNSentryEmitNewFrameEvent emitNewFrameEvent = ^(NSNumber *newFrameTimestampInSeconds) {}; + RNSentryEmitNewFrameEvent emitNewFrameEvent = ^(NSNumber *newFrameTimestampInSeconds) { }; [[RNSentryDependencyContainer sharedInstance] initializeFramesTrackerListenerWith:emitNewFrameEvent]; XCTAssertNotNil([[RNSentryDependencyContainer sharedInstance] framesTrackerListener]); diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.h b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.h index 3b3055e2f3..c987776703 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.h +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.h @@ -2,11 +2,6 @@ #import #import -@interface -SentrySDK (PrivateTests) -- (nullable SentryOptions *)options; -@end - @interface SentryDependencyContainer : NSObject + (instancetype)sharedInstance; @property (nonatomic, strong) SentryFramesTracker *framesTracker; diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.m b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.m index ee33d109e4..7a877795d6 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.m +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryFramesTrackerListenerTests.m @@ -46,7 +46,7 @@ - (void)testRNSentryFramesTrackerIsOneTimeListener OCMStub([(SentryDependencyContainer *)sentryDependencyContainerMock framesTracker]) .andReturn(frameTrackerMock); - RNSentryEmitNewFrameEvent mockEventEmitter = ^(NSNumber *newFrameTimestampInSeconds) {}; + RNSentryEmitNewFrameEvent mockEventEmitter = ^(NSNumber *newFrameTimestampInSeconds) { }; RNSentryFramesTrackerListener *actualListener = [[RNSentryFramesTrackerListener alloc] initWithSentryFramesTracker:[[SentryDependencyContainer sharedInstance] framesTracker] @@ -66,7 +66,7 @@ - (void)testRNSentryFramesTrackerAddsItselfAsListener OCMStub([(SentryDependencyContainer *)sentryDependencyContainerMock framesTracker]) .andReturn(frameTrackerMock); - RNSentryEmitNewFrameEvent mockEventEmitter = ^(NSNumber *newFrameTimestampInSeconds) {}; + RNSentryEmitNewFrameEvent mockEventEmitter = ^(NSNumber *newFrameTimestampInSeconds) { }; RNSentryFramesTrackerListener *actualListener = [[RNSentryFramesTrackerListener alloc] initWithSentryFramesTracker:[[SentryDependencyContainer sharedInstance] framesTracker] diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.h b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.h index 2ef701d215..8a9df3a94e 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.h +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.h @@ -1,8 +1,7 @@ #import "RNSentryOnDrawReporter.h" #import -@interface -RNSentryOnDrawReporterView (Testing) +@interface RNSentryOnDrawReporterView (Testing) + (instancetype)createWithMockedListener; - (RNSentryEmitNewFrameEvent)createEmitNewFrameEvent; diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.mm b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.mm index 3aca532855..da85363b16 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.mm +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryOnDrawReporter+Test.mm @@ -28,8 +28,7 @@ - (void)framesTrackerHasNewFrame:(nonnull NSDate *)newFrameDate @end -@implementation -RNSentryOnDrawReporterView (Testing) +@implementation RNSentryOnDrawReporterView (Testing) + (instancetype)createWithMockedListener { diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryReplayOptionsTests.swift b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryReplayOptionsTests.swift index 200d6422ec..5dcd2be3ec 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryReplayOptionsTests.swift +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryReplayOptionsTests.swift @@ -48,7 +48,7 @@ final class RNSentryReplayOptions: XCTestCase { } func assertAllDefaultReplayOptionsAreNotNil(replayOptions: [String: Any]) { - XCTAssertEqual(replayOptions.count, 8) + XCTAssertEqual(replayOptions.count, 9) XCTAssertNotNil(replayOptions["sessionSampleRate"]) XCTAssertNotNil(replayOptions["errorSampleRate"]) XCTAssertNotNil(replayOptions["maskAllImages"]) @@ -57,6 +57,7 @@ final class RNSentryReplayOptions: XCTestCase { XCTAssertNotNil(replayOptions["sdkInfo"]) XCTAssertNotNil(replayOptions["enableViewRendererV2"]) XCTAssertNotNil(replayOptions["enableFastViewRendering"]) + XCTAssertNotNil(replayOptions["quality"]) } func testSessionSampleRate() { @@ -66,7 +67,7 @@ final class RNSentryReplayOptions: XCTestCase { ] as NSDictionary).mutableCopy() as! NSMutableDictionary RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertEqual(actualOptions.sessionReplay.sessionSampleRate, 0.75) } @@ -77,7 +78,7 @@ final class RNSentryReplayOptions: XCTestCase { ] as NSDictionary).mutableCopy() as! NSMutableDictionary RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertEqual(actualOptions.sessionReplay.onErrorSampleRate, 0.75) } @@ -107,7 +108,7 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertEqual(actualOptions.sessionReplay.maskAllImages, true) assertContainsClass(classArray: actualOptions.sessionReplay.maskedViewClasses, stringClass: "RCTImageView") @@ -122,7 +123,7 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertEqual(actualOptions.sessionReplay.maskAllImages, false) XCTAssertEqual(actualOptions.sessionReplay.maskedViewClasses.count, 0) @@ -137,7 +138,7 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertEqual(actualOptions.sessionReplay.maskAllText, true) assertContainsClass(classArray: actualOptions.sessionReplay.maskedViewClasses, stringClass: "RCTTextView") @@ -161,13 +162,13 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertEqual(actualOptions.sessionReplay.maskAllText, false) XCTAssertEqual(actualOptions.sessionReplay.maskedViewClasses.count, 0) } - func testEnableViewRendererV2Default() { + func testEnableExperimentalViewRendererDefault() { let optionsDict = ([ "dsn": "https://abc@def.ingest.sentry.io/1234567", "replaysOnErrorSampleRate": 0.75 @@ -175,12 +176,12 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) - XCTAssertTrue(actualOptions.sessionReplay.enableViewRendererV2) + XCTAssertTrue(actualOptions.sessionReplay.enableExperimentalViewRenderer) } - func testEnableViewRendererV2True() { + func testEnableExperimentalViewRendererTrue() { let optionsDict = ([ "dsn": "https://abc@def.ingest.sentry.io/1234567", "replaysOnErrorSampleRate": 0.75, @@ -189,12 +190,12 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertTrue(actualOptions.sessionReplay.enableViewRendererV2) } - func testEnableViewRendererV2False() { + func testEnableExperimentalViewRendererFalse() { let optionsDict = ([ "dsn": "https://abc@def.ingest.sentry.io/1234567", "replaysOnErrorSampleRate": 0.75, @@ -203,7 +204,7 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertFalse(actualOptions.sessionReplay.enableViewRendererV2) } @@ -216,7 +217,7 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertFalse(actualOptions.sessionReplay.enableFastViewRendering) } @@ -230,7 +231,7 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertTrue(actualOptions.sessionReplay.enableFastViewRendering) } @@ -244,9 +245,77 @@ final class RNSentryReplayOptions: XCTestCase { RNSentryReplay.updateOptions(optionsDict) - let actualOptions = try! Options(dict: optionsDict as! [String: Any]) + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) XCTAssertFalse(actualOptions.sessionReplay.enableFastViewRendering) } + func testReplayQualityDefault() { + let optionsDict = ([ + "dsn": "https://abc@def.ingest.sentry.io/1234567", + "replaysOnErrorSampleRate": 0.75 + ] as NSDictionary).mutableCopy() as! NSMutableDictionary + + RNSentryReplay.updateOptions(optionsDict) + + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) + + XCTAssertEqual(actualOptions.sessionReplay.quality, SentryReplayOptions.SentryReplayQuality.medium) + } + + func testReplayQualityLow() { + let optionsDict = ([ + "dsn": "https://abc@def.ingest.sentry.io/1234567", + "replaysOnErrorSampleRate": 0.75, + "replaysSessionQuality": "low" + ] as NSDictionary).mutableCopy() as! NSMutableDictionary + + RNSentryReplay.updateOptions(optionsDict) + + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) + + XCTAssertEqual(actualOptions.sessionReplay.quality, SentryReplayOptions.SentryReplayQuality.low) + } + + func testReplayQualityMedium() { + let optionsDict = ([ + "dsn": "https://abc@def.ingest.sentry.io/1234567", + "replaysOnErrorSampleRate": 0.75, + "replaysSessionQuality": "medium" + ] as NSDictionary).mutableCopy() as! NSMutableDictionary + + RNSentryReplay.updateOptions(optionsDict) + + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) + + XCTAssertEqual(actualOptions.sessionReplay.quality, SentryReplayOptions.SentryReplayQuality.medium) + } + + func testReplayQualityHigh() { + let optionsDict = ([ + "dsn": "https://abc@def.ingest.sentry.io/1234567", + "replaysOnErrorSampleRate": 0.75, + "replaysSessionQuality": "high" + ] as NSDictionary).mutableCopy() as! NSMutableDictionary + + RNSentryReplay.updateOptions(optionsDict) + + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) + + XCTAssertEqual(actualOptions.sessionReplay.quality, SentryReplayOptions.SentryReplayQuality.high) + } + + func testReplayQualityInvalidFallsBackToMedium() { + let optionsDict = ([ + "dsn": "https://abc@def.ingest.sentry.io/1234567", + "replaysOnErrorSampleRate": 0.75, + "replaysSessionQuality": "invalid" + ] as NSDictionary).mutableCopy() as! NSMutableDictionary + + RNSentryReplay.updateOptions(optionsDict) + + let actualOptions = try! SentryOptionsInternal.initWithDict(optionsDict as! [String: Any]) + + XCTAssertEqual(actualOptions.sessionReplay.quality, SentryReplayOptions.SentryReplayQuality.medium) + } } diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.h b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.h index 8c2fddad03..43a25477fc 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.h +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.h @@ -1,26 +1,9 @@ #import #import -@interface -SentrySDK (PrivateTests) -- (nullable SentryOptions *)options; -@end - -@interface SentryBinaryImageInfo : NSObject -@property (nonatomic, strong) NSString *name; -@property (nonatomic) uint64_t address; -@property (nonatomic) uint64_t size; -@end +@class SentryOptions; -@interface SentryBinaryImageCache : NSObject -@property (nonatomic, readonly, class) SentryBinaryImageCache *shared; -- (void)start; -- (void)stop; -- (nullable SentryBinaryImageInfo *)imageByAddress:(const uint64_t)address; -@end +@interface SentrySDKInternal (PrivateTests) -@interface SentryDependencyContainer : NSObject -+ (instancetype)sharedInstance; -@property (nonatomic, strong) SentryDebugImageProvider *debugImageProvider; -@property (nonatomic, strong) SentryBinaryImageCache *binaryImageCache; ++ (nullable SentryOptions *)options; @end diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.mm b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.m similarity index 54% rename from packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.mm rename to packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.m index abe2ae70ce..677230eea3 100644 --- a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.mm +++ b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryTests.m @@ -3,9 +3,9 @@ #import #import #import -#import #import #import +@import Sentry; @interface RNSentryInitNativeSdkTests : XCTestCase @@ -227,6 +227,126 @@ - (void)testCreateOptionsWithDictionarySpotlightZero XCTAssertFalse(actualOptions.enableSpotlight, @"Did not disable spotlight"); } +- (void)testCreateOptionsWithDictionaryEnableUnhandledCPPExceptionsV2Enabled +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"_experiments" : @ { + @"enableUnhandledCPPExceptionsV2" : @YES, + }, + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableUnhandledCPPExceptionsV2"] boolValue]; + XCTAssertTrue( + enableUnhandledCPPExceptions, @"enableUnhandledCPPExceptionsV2 should be enabled"); +} + +- (void)testCreateOptionsWithDictionaryEnableUnhandledCPPExceptionsV2Disabled +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"_experiments" : @ { + @"enableUnhandledCPPExceptionsV2" : @NO, + }, + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableUnhandledCPPExceptionsV2"] boolValue]; + XCTAssertFalse( + enableUnhandledCPPExceptions, @"enableUnhandledCPPExceptionsV2 should be disabled"); +} + +- (void)testCreateOptionsWithDictionaryEnableUnhandledCPPExceptionsV2Default +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + // Test that when no _experiments are provided, the experimental option defaults to false + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableUnhandledCPPExceptionsV2"] boolValue]; + XCTAssertFalse( + enableUnhandledCPPExceptions, @"enableUnhandledCPPExceptionsV2 should default to disabled"); +} + +- (void)testCreateOptionsWithDictionaryEnableLogsEnabled +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"enableLogs" : @YES, + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableLogs = [[experimentalOptions valueForKey:@"enableLogs"] boolValue]; + XCTAssertTrue(enableLogs, @"enableLogs should be enabled"); +} + +- (void)testCreateOptionsWithDictionaryEnableLogsDisabled +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"enableLogs" : @NO, + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableLogs = [[experimentalOptions valueForKey:@"enableLogs"] boolValue]; + XCTAssertFalse(enableLogs, @"enableLogs should be disabled"); +} + - (void)testPassesErrorOnWrongDsn { NSError *error = nil; @@ -373,8 +493,8 @@ - (void)prepareNativeFrameMocksWithLocalSymbolication:(BOOL)debug SentryOptions *sentryOptions = [[SentryOptions alloc] init]; sentryOptions.debug = debug; // no local symbolication - id sentrySDKMock = OCMClassMock([SentrySDK class]); - OCMStub([(SentrySDK *)sentrySDKMock options]).andReturn(sentryOptions); + id sentrySDKMock = OCMClassMock([SentrySDKInternal class]); + OCMStub([(Class)sentrySDKMock options]).andReturn(sentryOptions); id sentryDependencyContainerMock = OCMClassMock([SentryDependencyContainer class]); OCMStub(ClassMethod([sentryDependencyContainerMock sharedInstance])) @@ -485,4 +605,244 @@ - (void)testFetchNativeStackFramesByInstructionsOnDeviceSymbolication XCTAssertTrue([actual isEqualToDictionary:expected]); } +- (void)testIgnoreErrorsDropsMatchingExceptionValue +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + NSDictionary *mockedOptions = @{ + @"dsn" : @"https://abc@def.ingest.sentry.io/1234567", + @"ignoreErrorsRegex" : @[ @"IgnoreMe.*" ] + }; + SentryOptions *options = [rnSentry createOptionsWithDictionary:mockedOptions error:&error]; + XCTAssertNotNil(options); + XCTAssertNil(error); + SentryEvent *event = [[SentryEvent alloc] init]; + SentryException *exception = [SentryException alloc]; + exception.value = @"IgnoreMe: This should be ignored"; + event.exceptions = @[ exception ]; + SentryEvent *result = options.beforeSend(event); + XCTAssertNil(result, @"Event with matching exception.value should be dropped"); +} + +- (void)testIgnoreErrorsDropsMatchingEventMessage +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + NSDictionary *mockedOptions = @{ + @"dsn" : @"https://abc@def.ingest.sentry.io/1234567", + @"ignoreErrorsStr" : @[ @"DropThisError" ] + }; + SentryOptions *options = [rnSentry createOptionsWithDictionary:mockedOptions error:&error]; + XCTAssertNotNil(options); + XCTAssertNil(error); + SentryEvent *event = [[SentryEvent alloc] init]; + SentryMessage *msg = [SentryMessage alloc]; + msg.message = @"DropThisError: should be dropped"; + event.message = msg; + SentryEvent *result = options.beforeSend(event); + XCTAssertNil(result, @"Event with matching event.message.formatted should be dropped"); +} + +- (void)testIgnoreErrorsDoesNotDropNonMatchingEvent +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + NSDictionary *mockedOptions = @{ + @"dsn" : @"https://abc@def.ingest.sentry.io/1234567", + @"ignoreErrorsRegex" : @[ @"IgnoreMe.*" ] + }; + SentryOptions *options = [rnSentry createOptionsWithDictionary:mockedOptions error:&error]; + XCTAssertNotNil(options); + XCTAssertNil(error); + SentryEvent *event = [[SentryEvent alloc] init]; + SentryException *exception = [SentryException alloc]; + exception.value = @"SomeOtherError: should not be ignored"; + event.exceptions = @[ exception ]; + SentryMessage *msg = [SentryMessage alloc]; + msg.message = @"SomeOtherMessage"; + event.message = msg; + SentryEvent *result = options.beforeSend(event); + XCTAssertNotNil(result, @"Event with non-matching error should not be dropped"); +} + +- (void)testIgnoreErrorsDropsMatchingExactString +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + NSDictionary *mockedOptions = @{ + @"dsn" : @"https://abc@def.ingest.sentry.io/1234567", + @"ignoreErrorsStr" : @[ @"ExactError" ] + }; + SentryOptions *options = [rnSentry createOptionsWithDictionary:mockedOptions error:&error]; + XCTAssertNotNil(options); + XCTAssertNil(error); + SentryEvent *event = [[SentryEvent alloc] init]; + SentryMessage *msg = [SentryMessage alloc]; + msg.message = @"ExactError"; + event.message = msg; + SentryEvent *result = options.beforeSend(event); + XCTAssertNil(result, @"Event with exactly matching string should be dropped"); +} + +- (void)testIgnoreErrorsRegexAndStringBothWork +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + NSDictionary *mockedOptions = @{ + @"dsn" : @"https://abc@def.ingest.sentry.io/1234567", + @"ignoreErrorsStr" : @[ @"ExactError" ], + @"ignoreErrorsRegex" : @[ @"IgnoreMe.*" ], + + }; + SentryOptions *options = [rnSentry createOptionsWithDictionary:mockedOptions error:&error]; + XCTAssertNotNil(options); + XCTAssertNil(error); + // Test regex match + SentryEvent *event1 = [[SentryEvent alloc] init]; + SentryException *exception = [SentryException alloc]; + exception.value = @"IgnoreMe: This should be ignored"; + event1.exceptions = @[ exception ]; + SentryEvent *result1 = options.beforeSend(event1); + XCTAssertNil(result1, @"Event with matching regex should be dropped"); + // Test exact string match + SentryEvent *event2 = [[SentryEvent alloc] init]; + SentryMessage *msg = [SentryMessage alloc]; + msg.message = @"ExactError"; + event2.message = msg; + SentryEvent *result2 = options.beforeSend(event2); + XCTAssertNil(result2, @"Event with exactly matching string should be dropped"); + // Test non-matching + SentryEvent *event3 = [[SentryEvent alloc] init]; + SentryMessage *msg3 = [SentryMessage alloc]; + msg3.message = @"OtherError"; + event3.message = msg3; + SentryEvent *result3 = options.beforeSend(event3); + XCTAssertNotNil(result3, @"Event with non-matching error should not be dropped"); +} + +- (void)testCreateOptionsWithDictionaryEnableSessionReplayInUnreliableEnvironmentDefault +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableSessionReplayInUnreliableEnvironment"] boolValue]; + XCTAssertFalse(enableUnhandledCPPExceptions, + @"enableSessionReplayInUnreliableEnvironment should be disabled"); +} + +- (void)testCreateOptionsWithDictionaryEnableSessionReplayInUnreliableEnvironmentWithErrorSampleRate +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"replaysOnErrorSampleRate" : @1.0, + @"replaysSessionSampleRate" : @0 + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableSessionReplayInUnreliableEnvironment"] boolValue]; + XCTAssertTrue(enableUnhandledCPPExceptions, + @"enableSessionReplayInUnreliableEnvironment should be enabled"); +} + +- (void) + testCreateOptionsWithDictionaryEnableSessionReplayInUnreliableEnvironmentWithSessionSampleRate +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"replaysOnErrorSampleRate" : @0.0, + @"replaysSessionSampleRate" : @0.1 + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableSessionReplayInUnreliableEnvironment"] boolValue]; + XCTAssertTrue(enableUnhandledCPPExceptions, + @"enableSessionReplayInUnreliableEnvironment should be enabled"); +} + +- (void) + testCreateOptionsWithDictionaryEnableSessionReplayInUnreliableEnvironmentWithSessionSampleRates +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"replaysOnErrorSampleRate" : @1.0, + @"replaysSessionSampleRate" : @0.1 + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableSessionReplayInUnreliableEnvironment"] boolValue]; + XCTAssertTrue(enableUnhandledCPPExceptions, + @"enableSessionReplayInUnreliableEnvironment should be enabled"); +} + +- (void)testCreateOptionsWithDictionaryEnableSessionReplayInUnreliableEnvironmentDisabled +{ + RNSentry *rnSentry = [[RNSentry alloc] init]; + NSError *error = nil; + + NSDictionary *_Nonnull mockedReactNativeDictionary = @{ + @"dsn" : @"https://abcd@efgh.ingest.sentry.io/123456", + @"replaysOnErrorSampleRate" : @0, + @"replaysSessionSampleRate" : @0 + }; + SentryOptions *actualOptions = [rnSentry createOptionsWithDictionary:mockedReactNativeDictionary + error:&error]; + + XCTAssertNotNil(actualOptions, @"Did not create sentry options"); + XCTAssertNil(error, @"Should not pass no error"); + + id experimentalOptions = [actualOptions valueForKey:@"experimental"]; + XCTAssertNotNil(experimentalOptions, @"Experimental options should not be nil"); + + BOOL enableUnhandledCPPExceptions = + [[experimentalOptions valueForKey:@"enableSessionReplayInUnreliableEnvironment"] boolValue]; + XCTAssertFalse(enableUnhandledCPPExceptions, + @"enableSessionReplayInUnreliableEnvironment should be disabled"); +} + @end diff --git a/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryUserTests.mm b/packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryUserTests.m similarity index 100% rename from packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryUserTests.mm rename to packages/core/RNSentryCocoaTester/RNSentryCocoaTesterTests/RNSentryUserTests.m diff --git a/packages/core/android/build.gradle b/packages/core/android/build.gradle index 292f7f6417..607a036ce5 100644 --- a/packages/core/android/build.gradle +++ b/packages/core/android/build.gradle @@ -53,6 +53,7 @@ android { } dependencies { + compileOnly files('libs/replay-stubs.jar') implementation 'com.facebook.react:react-native:+' - api 'io.sentry:sentry-android:7.22.5' + api 'io.sentry:sentry-android:8.23.0' } diff --git a/packages/core/android/libs/replay-stubs.jar b/packages/core/android/libs/replay-stubs.jar new file mode 100644 index 0000000000..116449310b Binary files /dev/null and b/packages/core/android/libs/replay-stubs.jar differ diff --git a/packages/core/android/replay-stubs/README.md b/packages/core/android/replay-stubs/README.md new file mode 100644 index 0000000000..fc3ea9404e --- /dev/null +++ b/packages/core/android/replay-stubs/README.md @@ -0,0 +1,17 @@ +This module is needed to successfully compile the Android target for the cases when `sentry-android-replay` is excluded from the host app classpath (for example, to reduce the resulting bundle/apk size), e.g. via the following snippet: + +```gradle +subprojects { + configurations.all { + exclude group: 'io.sentry', module: 'sentry-android-replay' + } +} +``` + +It provides stubs for the Replay classes that are used by the React Native SDK and is being added as a `compileOnly` dependency to `android/build.gradle` (meaning, it is not present at runtime and does not affect our customers' code). + +In addition, we also check for the `sentry-android-replay` classes presence at runtime and only then instantiate the replay-related classes (currently only `RNSentryReplayBreadcrumbConverter`) to not cause a `NoClassDefFoundError`. + +## Updating the stubs + +To update the stubs, just run `yarn build` from the root of the repo and it will recompile the classes and put them under `packages/core/android/libs/replay-stubs.jar`. Check this newly generated `.jar` in and push. diff --git a/packages/core/android/replay-stubs/build.gradle b/packages/core/android/replay-stubs/build.gradle new file mode 100644 index 0000000000..77a6c3a8b3 --- /dev/null +++ b/packages/core/android/replay-stubs/build.gradle @@ -0,0 +1,22 @@ +allprojects { + repositories { + mavenCentral() + } +} + +apply plugin: 'java-library' + +java { + sourceCompatibility = JavaVersion.VERSION_1_8 + targetCompatibility = JavaVersion.VERSION_1_8 +} + +tasks.named('jar', Jar) { + archiveBaseName.set('replay-stubs') + archiveVersion.set('') + destinationDirectory.set(file("$rootDir/../libs")) +} + +dependencies { + compileOnly 'io.sentry:sentry:8.23.0' +} diff --git a/packages/core/android/replay-stubs/gradle/wrapper/gradle-wrapper.jar b/packages/core/android/replay-stubs/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000000..9bbc975c74 Binary files /dev/null and b/packages/core/android/replay-stubs/gradle/wrapper/gradle-wrapper.jar differ diff --git a/packages/core/android/replay-stubs/gradle/wrapper/gradle-wrapper.properties b/packages/core/android/replay-stubs/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 0000000000..37f853b1c8 --- /dev/null +++ b/packages/core/android/replay-stubs/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,7 @@ +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip +networkTimeout=10000 +validateDistributionUrl=true +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists diff --git a/packages/core/android/replay-stubs/gradlew b/packages/core/android/replay-stubs/gradlew new file mode 100755 index 0000000000..faf93008b7 --- /dev/null +++ b/packages/core/android/replay-stubs/gradlew @@ -0,0 +1,251 @@ +#!/bin/sh + +# +# Copyright © 2015-2021 the original authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 +# + +############################################################################## +# +# Gradle start up script for POSIX generated by Gradle. +# +# Important for running: +# +# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is +# noncompliant, but you have some other compliant shell such as ksh or +# bash, then to run this script, type that shell name before the whole +# command line, like: +# +# ksh Gradle +# +# Busybox and similar reduced shells will NOT work, because this script +# requires all of these POSIX shell features: +# * functions; +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». +# +# Important for patching: +# +# (2) This script targets any POSIX shell, so it avoids extensions provided +# by Bash, Ksh, etc; in particular arrays are avoided. +# +# The "traditional" practice of packing multiple parameters into a +# space-separated string is a well documented source of bugs and security +# problems, so this is (mostly) avoided, by progressively accumulating +# options in "$@", and eventually passing that to Java. +# +# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, +# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; +# see the in-line comments for details. +# +# There are tweaks for specific operating systems such as AIX, CygWin, +# Darwin, MinGW, and NonStop. +# +# (3) This script is generated from the Groovy template +# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# within the Gradle project. +# +# You can find Gradle at https://github.com/gradle/gradle/. +# +############################################################################## + +# Attempt to set APP_HOME + +# Resolve links: $0 may be a link +app_path=$0 + +# Need this for daisy-chained symlinks. +while + APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path + [ -h "$app_path" ] +do + ls=$( ls -ld "$app_path" ) + link=${ls#*' -> '} + case $link in #( + /*) app_path=$link ;; #( + *) app_path=$APP_HOME$link ;; + esac +done + +# This is normally unused +# shellcheck disable=SC2034 +APP_BASE_NAME=${0##*/} +# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) +APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD=maximum + +warn () { + echo "$*" +} >&2 + +die () { + echo + echo "$*" + echo + exit 1 +} >&2 + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "$( uname )" in #( + CYGWIN* ) cygwin=true ;; #( + Darwin* ) darwin=true ;; #( + MSYS* | MINGW* ) msys=true ;; #( + NONSTOP* ) nonstop=true ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD=$JAVA_HOME/jre/sh/java + else + JAVACMD=$JAVA_HOME/bin/java + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD=java + if ! command -v java >/dev/null 2>&1 + then + die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +fi + +# Increase the maximum file descriptors if we can. +if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then + case $MAX_FD in #( + max*) + # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + MAX_FD=$( ulimit -H -n ) || + warn "Could not query maximum file descriptor limit" + esac + case $MAX_FD in #( + '' | soft) :;; #( + *) + # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + ulimit -n "$MAX_FD" || + warn "Could not set maximum file descriptor limit to $MAX_FD" + esac +fi + +# Collect all arguments for the java command, stacking in reverse order: +# * args from the command line +# * the main class name +# * -classpath +# * -D...appname settings +# * --module-path (only if needed) +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. + +# For Cygwin or MSYS, switch paths to Windows format before running java +if "$cygwin" || "$msys" ; then + APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) + CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) + + JAVACMD=$( cygpath --unix "$JAVACMD" ) + + # Now convert the arguments - kludge to limit ourselves to /bin/sh + for arg do + if + case $arg in #( + -*) false ;; # don't mess with options #( + /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath + [ -e "$t" ] ;; #( + *) false ;; + esac + then + arg=$( cygpath --path --ignore --mixed "$arg" ) + fi + # Roll the args list around exactly as many times as the number of + # args, so each arg winds up back in the position where it started, but + # possibly modified. + # + # NB: a `for` loop captures its iteration list before it begins, so + # changing the positional parameters here affects neither the number of + # iterations, nor the values presented in `arg`. + shift # remove old arg + set -- "$@" "$arg" # push replacement arg + done +fi + + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Collect all arguments for the java command: +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, +# and any embedded shellness will be escaped. +# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be +# treated as '${Hostname}' itself on the command line. + +set -- \ + "-Dorg.gradle.appname=$APP_BASE_NAME" \ + -classpath "$CLASSPATH" \ + org.gradle.wrapper.GradleWrapperMain \ + "$@" + +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + +# Use "xargs" to parse quoted args. +# +# With -n1 it outputs one arg per line, with the quotes and backslashes removed. +# +# In Bash we could simply go: +# +# readarray ARGS < <( xargs -n1 <<<"$var" ) && +# set -- "${ARGS[@]}" "$@" +# +# but POSIX shell has neither arrays nor command substitution, so instead we +# post-process each arg (as a line of input to sed) to backslash-escape any +# character that might be a shell metacharacter, then use eval to reverse +# that process (while maintaining the separation between arguments), and wrap +# the whole thing up as a single "set" statement. +# +# This will of course break if any of these variables contains a newline or +# an unmatched quote. +# + +eval "set -- $( + printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | + xargs -n1 | + sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | + tr '\n' ' ' + )" '"$@"' + +exec "$JAVACMD" "$@" diff --git a/packages/core/android/replay-stubs/gradlew.bat b/packages/core/android/replay-stubs/gradlew.bat new file mode 100644 index 0000000000..9b42019c79 --- /dev/null +++ b/packages/core/android/replay-stubs/gradlew.bat @@ -0,0 +1,94 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem +@rem SPDX-License-Identifier: Apache-2.0 +@rem + +@if "%DEBUG%"=="" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%"=="" set DIRNAME=. +@rem This is normally unused +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if %ERRORLEVEL% equ 0 goto execute + +echo. 1>&2 +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2 +echo. 1>&2 +echo Please set the JAVA_HOME variable in your environment to match the 1>&2 +echo location of your Java installation. 1>&2 + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto execute + +echo. 1>&2 +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2 +echo. 1>&2 +echo Please set the JAVA_HOME variable in your environment to match the 1>&2 +echo location of your Java installation. 1>&2 + +goto fail + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* + +:end +@rem End local scope for the variables with windows NT shell +if %ERRORLEVEL% equ 0 goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/packages/core/android/replay-stubs/settings.gradle b/packages/core/android/replay-stubs/settings.gradle new file mode 100644 index 0000000000..fddff8ac90 --- /dev/null +++ b/packages/core/android/replay-stubs/settings.gradle @@ -0,0 +1 @@ +include ':' diff --git a/packages/core/android/replay-stubs/src/main/java/io/sentry/android/replay/DefaultReplayBreadcrumbConverter.java b/packages/core/android/replay-stubs/src/main/java/io/sentry/android/replay/DefaultReplayBreadcrumbConverter.java new file mode 100644 index 0000000000..60023b7402 --- /dev/null +++ b/packages/core/android/replay-stubs/src/main/java/io/sentry/android/replay/DefaultReplayBreadcrumbConverter.java @@ -0,0 +1,13 @@ +package io.sentry.android.replay; + +import io.sentry.Breadcrumb; +import io.sentry.ReplayBreadcrumbConverter; +import io.sentry.rrweb.RRWebEvent; + +// just a stub to make the build pass when sentry-android-replay is not present +public class DefaultReplayBreadcrumbConverter implements ReplayBreadcrumbConverter { + @Override + public RRWebEvent convert(Breadcrumb breadcrumb) { + return null; + } +} 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 39e206d678..c0dfc4f005 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 @@ -29,17 +29,21 @@ import com.facebook.react.bridge.WritableNativeArray; import com.facebook.react.bridge.WritableNativeMap; import io.sentry.Breadcrumb; -import io.sentry.HubAdapter; import io.sentry.ILogger; import io.sentry.IScope; import io.sentry.ISentryExecutorService; import io.sentry.ISerializer; +import io.sentry.Integration; +import io.sentry.ScopesAdapter; import io.sentry.Sentry; import io.sentry.SentryDate; import io.sentry.SentryDateProvider; import io.sentry.SentryExecutorService; import io.sentry.SentryLevel; import io.sentry.SentryOptions; +import io.sentry.SentryReplayOptions; +import io.sentry.SentryReplayOptions.SentryReplayQuality; +import io.sentry.UncaughtExceptionHandlerIntegration; import io.sentry.android.core.AndroidLogger; import io.sentry.android.core.AndroidProfiler; import io.sentry.android.core.BuildInfoProvider; @@ -57,6 +61,7 @@ import io.sentry.util.DebugMetaPropertiesApplier; import io.sentry.util.FileUtils; import io.sentry.util.JsonSerializationUtils; +import io.sentry.util.LoadClass; import io.sentry.vendor.Base64; import java.io.BufferedInputStream; import java.io.BufferedReader; @@ -67,12 +72,16 @@ import java.io.IOException; import java.io.InputStream; import java.nio.charset.Charset; +import java.util.ArrayList; import java.util.HashMap; import java.util.Iterator; import java.util.List; +import java.util.Locale; import java.util.Map; import java.util.Properties; +import java.util.Set; import java.util.concurrent.CountDownLatch; +import java.util.regex.Pattern; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import org.jetbrains.annotations.TestOnly; @@ -120,12 +129,14 @@ public class RNSentryModuleImpl { private long maxTraceFileSize = 5 * 1024 * 1024; private final @NotNull SentryDateProvider dateProvider; + private final @NotNull LoadClass loadClass; public RNSentryModuleImpl(ReactApplicationContext reactApplicationContext) { packageInfo = getPackageInfo(reactApplicationContext); this.reactApplicationContext = reactApplicationContext; this.emitNewFrameEvent = createEmitNewFrameEvent(); this.dateProvider = new SentryAndroidDateProvider(); + this.loadClass = new LoadClass(); } private ReactApplicationContext getReactApplicationContext() { @@ -309,7 +320,7 @@ public void fetchNativeFrames(Promise promise) { } public void captureReplay(boolean isHardCrash, Promise promise) { - Sentry.getCurrentHub().getOptions().getReplayController().captureReplay(isHardCrash); + Sentry.getCurrentScopes().getOptions().getReplayController().captureReplay(isHardCrash); promise.resolve(getCurrentReplayId()); } @@ -405,7 +416,7 @@ public void fetchViewHierarchy(Promise promise) { return; } - ISerializer serializer = HubAdapter.getInstance().getOptions().getSerializer(); + ISerializer serializer = ScopesAdapter.getInstance().getOptions().getSerializer(); final @Nullable byte[] bytes = JsonSerializationUtils.bytesFrom(serializer, logger, viewHierarchy); if (bytes == null) { @@ -459,10 +470,6 @@ public void setUser(final ReadableMap userKeys, final ReadableMap userDataKeys) if (userKeys.hasKey("ip_address")) { userInstance.setIpAddress(userKeys.getString("ip_address")); } - - if (userKeys.hasKey("segment")) { - userInstance.setSegment(userKeys.getString("segment")); - } } if (userDataKeys != null) { @@ -624,8 +631,7 @@ private void initializeAndroidProfiler() { (int) SECONDS.toMicros(1) / profilingTracesHz, new SentryFrameMetricsCollector(reactApplicationContext, logger, buildInfo), executorService, - logger, - buildInfo); + logger); } public WritableMap startProfiling(boolean platformProfilers) { @@ -649,7 +655,7 @@ public WritableMap startProfiling(boolean platformProfilers) { } public WritableMap stopProfiling() { - final boolean isDebug = HubAdapter.getInstance().getOptions().isDebug(); + final boolean isDebug = ScopesAdapter.getInstance().getOptions().isDebug(); final WritableMap result = new WritableNativeMap(); File output = null; try { @@ -734,8 +740,15 @@ private String readStringFromFile(File path) throws IOException { } } + public void fetchNativeLogAttributes(Promise promise) { + final @NotNull SentryOptions options = ScopesAdapter.getInstance().getOptions(); + final @Nullable Context context = this.getReactApplicationContext().getApplicationContext(); + final @Nullable IScope currentScope = InternalSentrySdk.getCurrentScope(); + fetchNativeLogContexts(promise, options, context, currentScope); + } + public void fetchNativeDeviceContexts(Promise promise) { - final @NotNull SentryOptions options = HubAdapter.getInstance().getOptions(); + final @NotNull SentryOptions options = ScopesAdapter.getInstance().getOptions(); final @Nullable Context context = this.getReactApplicationContext().getApplicationContext(); final @Nullable IScope currentScope = InternalSentrySdk.getCurrentScope(); fetchNativeDeviceContexts(promise, options, context, currentScope); @@ -771,8 +784,50 @@ protected void fetchNativeDeviceContexts( promise.resolve(deviceContext); } + // Basically fetchNativeDeviceContexts but filtered to only get contexts info. + protected void fetchNativeLogContexts( + Promise promise, + final @NotNull SentryOptions options, + final @Nullable Context osContext, + final @Nullable IScope currentScope) { + if (!(options instanceof SentryAndroidOptions) || osContext == null) { + promise.resolve(null); + return; + } + + Object contextsObj = + InternalSentrySdk.serializeScope(osContext, (SentryAndroidOptions) options, currentScope) + .get("contexts"); + + if (!(contextsObj instanceof Map)) { + promise.resolve(null); + return; + } + + @SuppressWarnings("unchecked") + Map contextsMap = (Map) contextsObj; + + Map contextItems = new HashMap<>(); + if (contextsMap.containsKey("os")) { + contextItems.put("os", contextsMap.get("os")); + } + + if (contextsMap.containsKey("device")) { + contextItems.put("device", contextsMap.get("device")); + } + + contextItems.put("release", options.getRelease()); + + Map logContext = new HashMap<>(); + logContext.put("contexts", contextItems); + Object filteredContext = RNSentryMapConverter.convertToWritable(logContext); + + promise.resolve(filteredContext); + } + public void fetchNativeSdkInfo(Promise promise) { - final @Nullable SdkVersion sdkVersion = HubAdapter.getInstance().getOptions().getSdkVersion(); + final @Nullable SdkVersion sdkVersion = + ScopesAdapter.getInstance().getOptions().getSdkVersion(); if (sdkVersion == null) { promise.resolve(null); } else { diff --git a/packages/core/android/src/main/java/io/sentry/react/RNSentryStart.java b/packages/core/android/src/main/java/io/sentry/react/RNSentryStart.java index 86699ced05..3c41d04de5 100644 --- a/packages/core/android/src/main/java/io/sentry/react/RNSentryStart.java +++ b/packages/core/android/src/main/java/io/sentry/react/RNSentryStart.java @@ -147,6 +147,9 @@ static void getSentryAndroidOptions( if (rnOptions.hasKey("enableNdk")) { options.setEnableNdk(rnOptions.getBoolean("enableNdk")); } + if (rnOptions.hasKey("enableLogs")) { | ----------------------------------------------------------------------------------------------------------------------- + options.getLogs().setEnabled(rnOptions.getBoolean("enableLogs")); | ----------------------------------------------------------------------------------------------------------------------- + } if (rnOptions.hasKey("spotlight")) { if (rnOptions.getType("spotlight") == ReadableType.Boolean) { options.setEnableSpotlight(rnOptions.getBoolean("spotlight")); @@ -159,10 +162,16 @@ static void getSentryAndroidOptions( SentryReplayOptions replayOptions = getReplayOptions(rnOptions); options.setSessionReplay(replayOptions); - if (isReplayEnabled(replayOptions)) { + // Check if the replay integration is available on the classpath. It's already kept from R8 + // shrinking by sentry-android-core + final boolean isReplayAvailable = + loadClass.isClassAvailable("io.sentry.android.replay.ReplayIntegration", logger); + if (isReplayEnabled(replayOptions) && isReplayAvailable) { options.getReplayController().setBreadcrumbConverter(new RNSentryReplayBreadcrumbConverter()); } + trySetIgnoreErrors(options, rnOptions); + // Exclude Dev Server and Sentry Dsn request from Breadcrumbs String dsn = getURLFromDSN(rnOptions.getString("dsn")); String devServerUrl = rnOptions.getString("devServerUrl"); @@ -193,6 +202,38 @@ static void getSentryAndroidOptions( SentryLevel.INFO, String.format("Native Integrations '%s'", options.getIntegrations())); } + @TestOnly + protected void trySetIgnoreErrors(SentryAndroidOptions options, ReadableMap rnOptions) { + ReadableArray regErrors = null; + ReadableArray strErrors = null; + if (rnOptions.hasKey("ignoreErrorsRegex")) { + regErrors = rnOptions.getArray("ignoreErrorsRegex"); + } + if (rnOptions.hasKey("ignoreErrorsStr")) { + strErrors = rnOptions.getArray("ignoreErrorsStr"); + } + if (regErrors == null && strErrors == null) { + return; + } + + int regSize = regErrors != null ? regErrors.size() : 0; + int strSize = strErrors != null ? strErrors.size() : 0; + List list = new ArrayList<>(regSize + strSize); + if (regErrors != null) { + for (int i = 0; i < regErrors.size(); i++) { + list.add(regErrors.getString(i)); + } + } + if (strErrors != null) { + // Use the same behaviour of JavaScript instead of Android when dealing with strings. + for (int i = 0; i < strErrors.size(); i++) { + String pattern = ".*" + Pattern.quote(strErrors.getString(i)) + ".*"; + list.add(pattern); + } + } + options.setIgnoredErrors(list); + } + /** * This function updates the options with RNSentry defaults. These default can be overwritten by * users during manual native initialization. @@ -277,6 +318,12 @@ private static SentryReplayOptions getReplayOptions(@NotNull ReadableMap rnOptio ? rnOptions.getDouble("replaysOnErrorSampleRate") : null); + if (rnOptions.hasKey("replaysSessionQuality")) { + final String qualityString = rnOptions.getString("replaysSessionQuality"); + final SentryReplayQuality quality = parseReplayQuality(qualityString); + androidReplayOptions.setQuality(quality); + } + if (!rnOptions.hasKey("mobileReplayOptions")) { return androidReplayOptions; } @@ -350,6 +397,27 @@ private static void addPackages(SentryEvent event, SdkVersion sdk) { } } + private SentryReplayQuality parseReplayQuality(@Nullable String qualityString) { + if (qualityString == null) { + return SentryReplayQuality.MEDIUM; + } + + try { + switch (qualityString.toLowerCase(Locale.ROOT)) { + case "low": + return SentryReplayQuality.LOW; + case "medium": + return SentryReplayQuality.MEDIUM; + case "high": + return SentryReplayQuality.HIGH; + default: + return SentryReplayQuality.MEDIUM; + } + } catch (Exception e) { + return SentryReplayQuality.MEDIUM; + } + } + private static @Nullable String getURLFromDSN(@Nullable String dsn) { if (dsn == null) { return null; diff --git a/packages/core/android/src/main/java/io/sentry/react/RNSentryVersion.java b/packages/core/android/src/main/java/io/sentry/react/RNSentryVersion.java index a2527f94a8..8059753dd1 100644 --- a/packages/core/android/src/main/java/io/sentry/react/RNSentryVersion.java +++ b/packages/core/android/src/main/java/io/sentry/react/RNSentryVersion.java @@ -2,7 +2,7 @@ class RNSentryVersion { static final String REACT_NATIVE_SDK_PACKAGE_NAME = "npm:@sentry/react-native"; - static final String REACT_NATIVE_SDK_PACKAGE_VERSION = "6.15.1"; + static final String REACT_NATIVE_SDK_PACKAGE_VERSION = "7.4.0"; static final String NATIVE_SDK_NAME = "sentry.native.android.react-native"; static final String ANDROID_SDK_NAME = "sentry.java.android.react-native"; static final String REACT_NATIVE_SDK_NAME = "sentry.javascript.react-native"; 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 5b14f05c92..993969d830 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 @@ -127,6 +127,11 @@ public void disableNativeFramesTracking() { this.impl.disableNativeFramesTracking(); } + @Override + public void fetchNativeLogAttributes(Promise promise) { + this.impl.fetchNativeLogAttributes(promise); + } + @Override public void fetchNativeDeviceContexts(Promise promise) { this.impl.fetchNativeDeviceContexts(promise); diff --git a/packages/core/android/src/newarch/java/io/sentry/react/replay/RNSentryReplayUnmaskManager.java b/packages/core/android/src/newarch/java/io/sentry/react/replay/RNSentryReplayUnmaskManager.java index da0648123d..97b8358e5a 100644 --- a/packages/core/android/src/newarch/java/io/sentry/react/replay/RNSentryReplayUnmaskManager.java +++ b/packages/core/android/src/newarch/java/io/sentry/react/replay/RNSentryReplayUnmaskManager.java @@ -8,7 +8,7 @@ import com.facebook.react.viewmanagers.RNSentryReplayUnmaskManagerDelegate; import com.facebook.react.viewmanagers.RNSentryReplayUnmaskManagerInterface; -@ReactModule(name = RNSentryReplayMaskManagerImpl.REACT_CLASS) +@ReactModule(name = RNSentryReplayUnmaskManagerImpl.REACT_CLASS) public class RNSentryReplayUnmaskManager extends ViewGroupManager implements RNSentryReplayUnmaskManagerInterface { private final RNSentryReplayUnmaskManagerDelegate< @@ -23,7 +23,7 @@ public ViewManagerDelegate getDelegate() { @NonNull @Override public String getName() { - return RNSentryReplayMaskManagerImpl.REACT_CLASS; + return RNSentryReplayUnmaskManagerImpl.REACT_CLASS; } @NonNull diff --git a/packages/core/ios/RNSentry+fetchNativeStack.m b/packages/core/ios/RNSentry+fetchNativeStack.m new file mode 100644 index 0000000000..95a0cdf02f --- /dev/null +++ b/packages/core/ios/RNSentry+fetchNativeStack.m @@ -0,0 +1,96 @@ +#import "RNSentry.h" +#import "RNSentryBreadcrumb.h" +#import "RNSentryId.h" +#import +#import +#import +#import +#import +#import +#import +#import +#import +#import +@import Sentry; + +// This method was moved to a new category so we can use `@import Sentry` to use Sentry's Swift +// classes +@implementation RNSentry (fetchNativeStack) + +- (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAddr + symbolicate:(SymbolicateCallbackType)symbolicate +{ + BOOL shouldSymbolicateLocally = [SentrySDKInternal.options debug]; + + NSString *appPackageName = [[NSBundle mainBundle] executablePath]; + + NSMutableSet *_Nonnull imagesAddrToRetrieveDebugMetaImages = + [[NSMutableSet alloc] init]; + NSMutableArray *> *_Nonnull serializedFrames = + [[NSMutableArray alloc] init]; + + for (NSNumber *addr in instructionsAddr) { + SentryBinaryImageInfo *_Nullable image = [[[SentryDependencyContainer sharedInstance] + binaryImageCache] imageByAddress:[addr unsignedLongLongValue]]; + if (image != nil) { + NSString *imageAddr = sentry_formatHexAddressUInt64([image address]); + [imagesAddrToRetrieveDebugMetaImages addObject:imageAddr]; + + NSDictionary *_Nonnull nativeFrame = @{ + @"platform" : @"cocoa", + @"instruction_addr" : sentry_formatHexAddress(addr), + @"package" : [image name], + @"image_addr" : imageAddr, + @"in_app" : [NSNumber numberWithBool:[appPackageName isEqualToString:[image name]]], + }; + + if (shouldSymbolicateLocally) { + Dl_info symbolsBuffer; + bool symbols_succeed = false; + symbols_succeed + = symbolicate((void *)[addr unsignedLongLongValue], &symbolsBuffer) != 0; + if (symbols_succeed) { + NSMutableDictionary *_Nonnull symbolicated + = nativeFrame.mutableCopy; + symbolicated[@"symbol_addr"] + = sentry_formatHexAddressUInt64((uintptr_t)symbolsBuffer.dli_saddr); + symbolicated[@"function"] = [NSString stringWithCString:symbolsBuffer.dli_sname + encoding:NSUTF8StringEncoding]; + + nativeFrame = symbolicated; + } + } + + [serializedFrames addObject:nativeFrame]; + } else { + [serializedFrames addObject:@{ + @"platform" : @"cocoa", + @"instruction_addr" : sentry_formatHexAddress(addr), + }]; + } + } + + if (shouldSymbolicateLocally) { + return @{ + @"frames" : serializedFrames, + }; + } else { + NSMutableArray *> *_Nonnull serializedDebugMetaImages = + [[NSMutableArray alloc] init]; + + NSArray *debugMetaImages = + [[[SentryDependencyContainer sharedInstance] debugImageProvider] + getDebugImagesForImageAddressesFromCache:imagesAddrToRetrieveDebugMetaImages]; + + for (SentryDebugMeta *debugImage in debugMetaImages) { + [serializedDebugMetaImages addObject:[debugImage serialize]]; + } + + return @{ + @"frames" : serializedFrames, + @"debugMetaImages" : serializedDebugMetaImages, + }; + } +} + +@end diff --git a/packages/core/ios/RNSentry.h b/packages/core/ios/RNSentry.h index c7fb93e0ea..72d34b9a17 100644 --- a/packages/core/ios/RNSentry.h +++ b/packages/core/ios/RNSentry.h @@ -8,21 +8,25 @@ #import #import -#import -#import // This import exposes public RNSentrySDK start #import "RNSentrySDK.h" typedef int (*SymbolicateCallbackType)(const void *, Dl_info *); -@interface -SentrySDK (Private) +@class SentryOptions; +@class SentryEvent; + +@interface SentrySDKInternal : NSObject @property (nonatomic, nullable, readonly, class) SentryOptions *options; @end @interface RNSentry : RCTEventEmitter +@end + +@interface RNSentry (fetchNativeStack) + - (NSDictionary *_Nonnull)fetchNativeStackFramesBy:(NSArray *)instructionsAddr symbolicate:(SymbolicateCallbackType)symbolicate; diff --git a/packages/core/ios/RNSentry.mm b/packages/core/ios/RNSentry.mm index 7b479e1d53..117e49875b 100644 --- a/packages/core/ios/RNSentry.mm +++ b/packages/core/ios/RNSentry.mm @@ -20,12 +20,16 @@ #import "RNSentryId.h" #import #import -#import -#import +#import +#import #import +#import +#import #import -#import +#import +#import #import +#import // This guard prevents importing Hermes in JSC apps #if SENTRY_PROFILING_ENABLED @@ -50,22 +54,17 @@ #endif #import "RNSentryStart.h" +#import "RNSentryExperimentalOptions.h" #import "RNSentryVersion.h" - -@interface -SentrySDK (RNSentry) - -+ (void)captureEnvelope:(SentryEnvelope *)envelope; - -+ (void)storeEnvelope:(SentryEnvelope *)envelope; - -@end +#import "SentrySDKWrapper.h" static bool hasFetchedAppStart; @implementation RNSentry { bool hasListeners; RNSentryTimeToDisplay *_timeToDisplay; + NSArray *_ignoreErrorPatternsStr; + NSArray *_ignoreErrorPatternsRegex; } - (dispatch_queue_t)methodQueue @@ -87,11 +86,8 @@ - (instancetype)init } RCT_EXPORT_MODULE() - -RCT_EXPORT_METHOD(initNativeSdk - : (NSDictionary *_Nonnull)options resolve - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(initNativeSdk : (NSDictionary *_Nonnull)options resolve : ( + RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { NSError *error = nil; [RNSentryStart startWithOptions:options error:&error]; @@ -99,12 +95,222 @@ - (instancetype)init reject(@"SentryReactNative", error.localizedDescription, error); return; } + resolve(@YES); } RCT_EXPORT_METHOD(initNativeReactNavigationNewFrameTracking : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) +======= + + NSString *sdkVersion = [PrivateSentrySDKOnly getSdkVersionString]; + [PrivateSentrySDKOnly setSdkName:NATIVE_SDK_NAME andVersionString:sdkVersion]; + [PrivateSentrySDKOnly addSdkPackage:REACT_NATIVE_SDK_PACKAGE_NAME + version:REACT_NATIVE_SDK_PACKAGE_VERSION]; + + [SentrySDKWrapper startWithOptions:sentryOptions]; + +#if TARGET_OS_IPHONE || TARGET_OS_MACCATALYST + BOOL appIsActive = + [[UIApplication sharedApplication] applicationState] == UIApplicationStateActive; +#else + BOOL appIsActive = [[NSApplication sharedApplication] isActive]; +#endif + + // If the app is active/in foreground, and we have not sent the SentryHybridSdkDidBecomeActive + // notification, send it. + if (appIsActive && !sentHybridSdkDidBecomeActive + && (PrivateSentrySDKOnly.options.enableAutoSessionTracking + || PrivateSentrySDKOnly.options.enableWatchdogTerminationTracking)) { + [[NSNotificationCenter defaultCenter] postNotificationName:@"SentryHybridSdkDidBecomeActive" + object:nil]; + + sentHybridSdkDidBecomeActive = true; + } + +#if SENTRY_TARGET_REPLAY_SUPPORTED + [RNSentryReplay postInit]; +#endif + + resolve(@YES); +} +<<<<<<< +- (SentryOptions *_Nullable)createOptionsWithDictionary:(NSDictionary *_Nonnull)options + error:(NSError *_Nonnull *_Nonnull)errorPointer +{ + SentryBeforeSendEventCallback beforeSend = ^SentryEvent *(SentryEvent *event) { + // We don't want to send an event after startup that came from a Unhandled JS Exception of + // React Native because we sent it already before the app crashed. + if (nil != event.exceptions.firstObject.type && + [event.exceptions.firstObject.type rangeOfString:@"Unhandled JS Exception"].location + != NSNotFound) { + return nil; + } + + // Regex and Str are set when one of them has value so we only need to check one of them. + if (self->_ignoreErrorPatternsStr || self->_ignoreErrorPatternsRegex) { + for (SentryException *exception in event.exceptions) { + if ([self shouldIgnoreError:exception.value]) { + return nil; + } + } + if ([self shouldIgnoreError:event.message.message]) { + return nil; + } + } + + [self setEventOriginTag:event]; + return event; + }; + + NSMutableDictionary *mutableOptions = [options mutableCopy]; + [mutableOptions setValue:beforeSend forKey:@"beforeSend"]; + + // remove performance traces sample rate and traces sampler since we don't want to synchronize + // these configurations to the Native SDKs. The user could tho initialize the SDK manually and + // set themselves. + [mutableOptions removeObjectForKey:@"tracesSampleRate"]; + [mutableOptions removeObjectForKey:@"tracesSampler"]; + [mutableOptions removeObjectForKey:@"enableTracing"]; + +#if SENTRY_TARGET_REPLAY_SUPPORTED + BOOL isSessionReplayEnabled = [RNSentryReplay updateOptions:mutableOptions]; +#else + // Defaulting to false for unsupported targets + BOOL isSessionReplayEnabled = NO; +#endif + + SentryOptions *sentryOptions = [SentryOptionsInternal initWithDict:mutableOptions + didFailWithError:errorPointer]; + if (*errorPointer != nil) { + return nil; + } + + // Exclude Dev Server and Sentry Dsn request from Breadcrumbs + NSString *dsn = [self getURLFromDSN:[mutableOptions valueForKey:@"dsn"]]; + NSString *devServerUrl = [mutableOptions valueForKey:@"devServerUrl"]; + sentryOptions.beforeBreadcrumb + = ^SentryBreadcrumb *_Nullable(SentryBreadcrumb *_Nonnull breadcrumb) + { + NSString *url = breadcrumb.data[@"url"] ?: @""; + + if ([@"http" isEqualToString:breadcrumb.type] + && ((dsn != nil && [url hasPrefix:dsn]) + || (devServerUrl != nil && [url hasPrefix:devServerUrl]))) { + return nil; + } + return breadcrumb; + }; + + if ([mutableOptions valueForKey:@"enableNativeCrashHandling"] != nil) { + BOOL enableNativeCrashHandling = [mutableOptions[@"enableNativeCrashHandling"] boolValue]; + + if (!enableNativeCrashHandling) { + NSMutableArray *integrations = sentryOptions.integrations.mutableCopy; + [integrations removeObject:@"SentryCrashIntegration"]; + sentryOptions.integrations = integrations; + } + } + + // Set spotlight option + if ([mutableOptions valueForKey:@"spotlight"] != nil) { + id spotlightValue = [mutableOptions valueForKey:@"spotlight"]; + if ([spotlightValue isKindOfClass:[NSString class]]) { + NSLog(@"Using Spotlight on address: %@", spotlightValue); + sentryOptions.enableSpotlight = true; + sentryOptions.spotlightUrl = spotlightValue; + } else if ([spotlightValue isKindOfClass:[NSNumber class]]) { + sentryOptions.enableSpotlight = [spotlightValue boolValue]; + id defaultSpotlightUrl = [mutableOptions valueForKey:@"defaultSidecarUrl"]; + if (defaultSpotlightUrl != nil) { + sentryOptions.spotlightUrl = defaultSpotlightUrl; + } + } + } + + if ([mutableOptions valueForKey:@"enableLogs"] != nil) { + id enableLogsValue = [mutableOptions valueForKey:@"enableLogs"]; + if ([enableLogsValue isKindOfClass:[NSNumber class]]) { + [RNSentryExperimentalOptions setEnableLogs:[enableLogsValue boolValue] + sentryOptions:sentryOptions]; + } + } + [self trySetIgnoreErrors:mutableOptions]; + + // Enable the App start and Frames tracking measurements + if ([mutableOptions valueForKey:@"enableAutoPerformanceTracing"] != nil) { + BOOL enableAutoPerformanceTracing = + [mutableOptions[@"enableAutoPerformanceTracing"] boolValue]; + PrivateSentrySDKOnly.appStartMeasurementHybridSDKMode = enableAutoPerformanceTracing; +#if TARGET_OS_IPHONE || TARGET_OS_MACCATALYST + PrivateSentrySDKOnly.framesTrackingMeasurementHybridSDKMode = enableAutoPerformanceTracing; +#endif + } + + // Failed requests can only be enabled in one SDK to avoid duplicates + sentryOptions.enableCaptureFailedRequests = NO; + + NSDictionary *experiments = options[@"_experiments"]; + if (experiments != nil && [experiments isKindOfClass:[NSDictionary class]]) { + BOOL enableUnhandledCPPExceptions = + [experiments[@"enableUnhandledCPPExceptionsV2"] boolValue]; + [RNSentryExperimentalOptions setEnableUnhandledCPPExceptionsV2:enableUnhandledCPPExceptions + sentryOptions:sentryOptions]; + } + + if (isSessionReplayEnabled) { + [RNSentryExperimentalOptions setEnableSessionReplayInUnreliableEnvironment:YES + sentryOptions:sentryOptions]; + } + + return sentryOptions; +} + +- (NSString *_Nullable)getURLFromDSN:(NSString *)dsn +{ + NSURL *url = [NSURL URLWithString:dsn]; + if (!url) { + return nil; + } + return [NSString stringWithFormat:@"%@://%@", url.scheme, url.host]; +} + +- (void)setEventOriginTag:(SentryEvent *)event +{ + if (event.sdk != nil) { + NSString *sdkName = event.sdk[@"name"]; + + // If the event is from react native, it gets set + // there and we do not handle it here. + if ([sdkName isEqual:NATIVE_SDK_NAME]) { + [self setEventEnvironmentTag:event origin:@"ios" environment:@"native"]; + } + } +} + +- (void)setEventEnvironmentTag:(SentryEvent *)event + origin:(NSString *)origin + environment:(NSString *)environment +{ + NSMutableDictionary *newTags = [NSMutableDictionary new]; + + if (nil != event.tags && [event.tags count] > 0) { + [newTags addEntriesFromDictionary:event.tags]; + } + if (nil != origin) { + [newTags setValue:origin forKey:@"event.origin"]; + } + if (nil != environment) { + [newTags setValue:environment forKey:@"event.environment"]; + } + + event.tags = newTags; +} + +RCT_EXPORT_METHOD(initNativeReactNavigationNewFrameTracking : ( + RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) +>>>>>>> main { #if SENTRY_HAS_UIKIT if ([[NSThread currentThread] isMainThread]) { @@ -146,9 +352,8 @@ - (void)stopObserving return @[ RNSentryNewFrameEvent ]; } -RCT_EXPORT_METHOD(fetchNativeSdkInfo - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + fetchNativeSdkInfo : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { resolve(@ { @"name" : PrivateSentrySDKOnly.getSdkName, @@ -156,9 +361,8 @@ - (void)stopObserving }); } -RCT_EXPORT_METHOD(fetchModules - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + fetchModules : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { NSString *filePath = [[NSBundle mainBundle] pathForResource:@"modules" ofType:@"json"]; NSString *modulesString = [NSString stringWithContentsOfFile:filePath @@ -173,90 +377,65 @@ - (void)stopObserving return packageName; } -- (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAddr - symbolicate:(SymbolicateCallbackType)symbolicate -{ - BOOL shouldSymbolicateLocally = [SentrySDK.options debug]; - NSString *appPackageName = [[NSBundle mainBundle] executablePath]; - - NSMutableSet *_Nonnull imagesAddrToRetrieveDebugMetaImages = - [[NSMutableSet alloc] init]; - NSMutableArray *> *_Nonnull serializedFrames = - [[NSMutableArray alloc] init]; - - for (NSNumber *addr in instructionsAddr) { - SentryBinaryImageInfo *_Nullable image = [[[SentryDependencyContainer sharedInstance] - binaryImageCache] imageByAddress:[addr unsignedLongLongValue]]; - if (image != nil) { - NSString *imageAddr = sentry_formatHexAddressUInt64([image address]); - [imagesAddrToRetrieveDebugMetaImages addObject:imageAddr]; - - NSDictionary *_Nonnull nativeFrame = @{ - @"platform" : @"cocoa", - @"instruction_addr" : sentry_formatHexAddress(addr), - @"package" : [image name], - @"image_addr" : imageAddr, - @"in_app" : [NSNumber numberWithBool:[appPackageName isEqualToString:[image name]]], - }; - - if (shouldSymbolicateLocally) { - Dl_info symbolsBuffer; - bool symbols_succeed = false; - symbols_succeed - = symbolicate((void *)[addr unsignedLongLongValue], &symbolsBuffer) != 0; - if (symbols_succeed) { - NSMutableDictionary *_Nonnull symbolicated - = nativeFrame.mutableCopy; - symbolicated[@"symbol_addr"] - = sentry_formatHexAddressUInt64((uintptr_t)symbolsBuffer.dli_saddr); - symbolicated[@"function"] = [NSString stringWithCString:symbolsBuffer.dli_sname - encoding:NSUTF8StringEncoding]; - - nativeFrame = symbolicated; - } - } +RCT_EXPORT_SYNCHRONOUS_TYPED_METHOD( + NSDictionary *, fetchNativeStackFramesBy : (NSArray *)instructionsAddr) +{ + return [self fetchNativeStackFramesBy:instructionsAddr symbolicate:dladdr]; +} - [serializedFrames addObject:nativeFrame]; - } else { - [serializedFrames addObject:@{ - @"platform" : @"cocoa", - @"instruction_addr" : sentry_formatHexAddress(addr), - }]; - } - } +RCT_EXPORT_METHOD(fetchNativeLogAttributes : (RCTPromiseResolveBlock)resolve rejecter : ( + RCTPromiseRejectBlock)reject) +{ + __block NSMutableDictionary *result = [NSMutableDictionary new]; - if (shouldSymbolicateLocally) { - return @{ - @"frames" : serializedFrames, - }; - } else { - NSMutableArray *> *_Nonnull serializedDebugMetaImages = - [[NSMutableArray alloc] init]; + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { + // Serialize to get contexts dictionary + NSDictionary *serializedScope = [scope serialize]; + NSDictionary *allContexts = serializedScope[@"context"]; // It's singular here, annoyingly - NSArray *debugMetaImages = - [[[SentryDependencyContainer sharedInstance] debugImageProvider] - getDebugImagesForImageAddressesFromCache:imagesAddrToRetrieveDebugMetaImages]; + NSMutableDictionary *contexts = [NSMutableDictionary new]; - for (SentryDebugMeta *debugImage in debugMetaImages) { - [serializedDebugMetaImages addObject:[debugImage serialize]]; + NSDictionary *device = allContexts[@"device"]; + if ([device isKindOfClass:[NSDictionary class]]) { + contexts[@"device"] = device; } - return @{ - @"frames" : serializedFrames, - @"debugMetaImages" : serializedDebugMetaImages, - }; - } -} + NSDictionary *os = allContexts[@"os"]; + if ([os isKindOfClass:[NSDictionary class]]) { + contexts[@"os"] = os; + } -RCT_EXPORT_SYNCHRONOUS_TYPED_METHOD(NSDictionary *, fetchNativeStackFramesBy - : (NSArray *)instructionsAddr) -{ - return [self fetchNativeStackFramesBy:instructionsAddr symbolicate:dladdr]; + NSString *releaseName = SentrySDKInternal.options.releaseName; + if (releaseName) { + contexts[@"release"] = releaseName; + } + // Merge extra context + NSDictionary *extraContext = [PrivateSentrySDKOnly getExtraContext]; + + if (extraContext) { + NSDictionary *extraDevice = extraContext[@"device"]; + if ([extraDevice isKindOfClass:[NSDictionary class]]) { + NSMutableDictionary *mergedDevice = + [contexts[@"device"] mutableCopy] ?: [NSMutableDictionary new]; + [mergedDevice addEntriesFromDictionary:extraDevice]; + contexts[@"device"] = mergedDevice; + } + + NSDictionary *extraOS = extraContext[@"os"]; + if ([extraOS isKindOfClass:[NSDictionary class]]) { + NSMutableDictionary *mergedOS = + [contexts[@"os"] mutableCopy] ?: [NSMutableDictionary new]; + [mergedOS addEntriesFromDictionary:extraOS]; + contexts[@"os"] = mergedOS; + } + } + result[@"contexts"] = contexts; + }]; + resolve(result); } -RCT_EXPORT_METHOD(fetchNativeDeviceContexts - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(fetchNativeDeviceContexts : (RCTPromiseResolveBlock)resolve rejecter : ( + RCTPromiseRejectBlock)reject) { if (PrivateSentrySDKOnly.options.debug) { NSLog(@"Bridge call to: deviceContexts"); @@ -264,7 +443,7 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd __block NSMutableDictionary *serializedScope; // Temp work around until sorted out this API in sentry-cocoa. // TODO: If the callback isnt' executed the promise wouldn't be resolved. - [SentrySDK configureScope:^(SentryScope *_Nonnull scope) { + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { serializedScope = [[scope serialize] mutableCopy]; NSDictionary *user = [serializedScope valueForKey:@"user"]; @@ -316,9 +495,8 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd resolve(serializedScope); } -RCT_EXPORT_METHOD(fetchNativeAppStart - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + fetchNativeAppStart : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { #if SENTRY_HAS_UIKIT NSDictionary *measurements = @@ -343,9 +521,8 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd #endif } -RCT_EXPORT_METHOD(fetchNativeFrames - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + fetchNativeFrames : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { #if TARGET_OS_IPHONE || TARGET_OS_MACCATALYST @@ -374,9 +551,8 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd #endif } -RCT_EXPORT_METHOD(fetchNativeRelease - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + fetchNativeRelease : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { NSDictionary *infoDict = [[NSBundle mainBundle] infoDictionary]; resolve(@ { @@ -386,11 +562,8 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd }); } -RCT_EXPORT_METHOD(captureEnvelope - : (NSString *_Nonnull)rawBytes options - : (NSDictionary *_Nonnull)options resolve - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(captureEnvelope : (NSString *_Nonnull)rawBytes options : (NSDictionary *_Nonnull) + options resolve : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { NSData *data = [[NSData alloc] initWithBase64EncodedString:rawBytes options:0]; @@ -413,9 +586,8 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd resolve(@YES); } -RCT_EXPORT_METHOD(captureScreenshot - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + captureScreenshot : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { #if TARGET_OS_IPHONE || TARGET_OS_MACCATALYST NSArray *rawScreenshots = [PrivateSentrySDKOnly captureScreenshots]; @@ -447,9 +619,8 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd #endif } -RCT_EXPORT_METHOD(fetchViewHierarchy - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + fetchViewHierarchy : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { #if TARGET_OS_IPHONE || TARGET_OS_MACCATALYST NSData *rawViewHierarchy = [PrivateSentrySDKOnly captureViewHierarchy]; @@ -468,7 +639,7 @@ - (NSDictionary *)fetchNativeStackFramesBy:(NSArray *)instructionsAd RCT_EXPORT_METHOD(setUser : (NSDictionary *)userKeys otherUserKeys : (NSDictionary *)userDataKeys) { - [SentrySDK configureScope:^(SentryScope *_Nonnull scope) { + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { [scope setUser:[RNSentry userFrom:userKeys otherUserKeys:userDataKeys]]; }]; } @@ -517,7 +688,7 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys RCT_EXPORT_METHOD(addBreadcrumb : (NSDictionary *)breadcrumb) { - [SentrySDK configureScope:^(SentryScope *_Nonnull scope) { + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { [scope addBreadcrumb:[RNSentryBreadcrumb from:breadcrumb]]; }]; @@ -531,12 +702,12 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys RCT_EXPORT_METHOD(clearBreadcrumbs) { - [SentrySDK configureScope:^(SentryScope *_Nonnull scope) { [scope clearBreadcrumbs]; }]; + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { [scope clearBreadcrumbs]; }]; } RCT_EXPORT_METHOD(setExtra : (NSString *)key extra : (NSString *)extra) { - [SentrySDK + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { [scope setExtraValue:extra forKey:key]; }]; } @@ -546,7 +717,7 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys return; } - [SentrySDK configureScope:^(SentryScope *_Nonnull scope) { + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { if (context == nil) { [scope removeContextForKey:key]; } else { @@ -557,17 +728,16 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys RCT_EXPORT_METHOD(setTag : (NSString *)key value : (NSString *)value) { - [SentrySDK + [SentrySDKWrapper configureScope:^(SentryScope *_Nonnull scope) { [scope setTagValue:value forKey:key]; }]; } -RCT_EXPORT_METHOD(crash) { [SentrySDK crash]; } +RCT_EXPORT_METHOD(crash) { [SentrySDKWrapper crash]; } -RCT_EXPORT_METHOD(closeNativeSdk - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + closeNativeSdk : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { - [SentrySDK close]; + [SentrySDKWrapper close]; resolve(@YES); } @@ -584,10 +754,8 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys // the 'tracesSampleRate' or 'tracesSampler' option. } -RCT_EXPORT_METHOD(captureReplay - : (BOOL)isHardCrash resolver - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(captureReplay : (BOOL)isHardCrash resolver : ( + RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { #if SENTRY_TARGET_REPLAY_SUPPORTED [PrivateSentrySDKOnly captureReplay]; @@ -597,10 +765,8 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys #endif } -RCT_EXPORT_METHOD(getDataFromUri - : (NSString *_Nonnull)uri resolve - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(getDataFromUri : (NSString *_Nonnull)uri resolve : ( + RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { #if TARGET_OS_IPHONE || TARGET_OS_MACCATALYST NSURL *fileURL = [NSURL URLWithString:uri]; @@ -643,7 +809,15 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys { #if SENTRY_PROFILING_ENABLED try { +# ifdef NEW_HERMES_RUNTIME + auto *hermesAPI = facebook::jsi::castInterface( + facebook::hermes::makeHermesRootAPI()); + if (hermesAPI) { + hermesAPI->enableSamplingProfiler(); + } +# else facebook::hermes::HermesRuntime::enableSamplingProfiler(); +# endif if (nativeProfileTraceId == nil && nativeProfileStartTime == 0 && platformProfilers) { # if SENTRY_TARGET_PROFILING_SUPPORTED nativeProfileTraceId = [RNSentryId newId]; @@ -703,10 +877,19 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys nativeProfileTraceId = nil; nativeProfileStartTime = 0; - facebook::hermes::HermesRuntime::disableSamplingProfiler(); std::stringstream ss; +# ifdef NEW_HERMES_RUNTIME + auto *hermesAPI = facebook::jsi::castInterface( + facebook::hermes::makeHermesRootAPI()); + if (hermesAPI) { + hermesAPI->disableSamplingProfiler(); + hermesAPI->dumpSampledTraceToStream(ss); + } +# else + facebook::hermes::HermesRuntime::disableSamplingProfiler(); // Before RN 0.69 Hermes used llvh::raw_ostream (profiling is supported for 0.69 and newer) facebook::hermes::HermesRuntime::dumpSampledTraceToStream(ss); +# endif std::string s = ss.str(); NSString *data = [NSString stringWithCString:s.c_str() @@ -766,11 +949,10 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys #endif } -RCT_EXPORT_METHOD(crashedLastRun - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD( + crashedLastRun : (RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { - resolve(@([SentrySDK crashedLastRun])); + resolve(@([SentrySDKWrapper crashedLastRun])); } // Thanks to this guard, we won't compile this code when we build for the old architecture. @@ -782,17 +964,14 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys } #endif -RCT_EXPORT_METHOD(getNewScreenTimeToDisplay - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(getNewScreenTimeToDisplay : (RCTPromiseResolveBlock)resolve rejecter : ( + RCTPromiseRejectBlock)reject) { [_timeToDisplay getTimeToDisplay:resolve]; } -RCT_EXPORT_METHOD(popTimeToDisplayFor - : (NSString *)key resolver - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(popTimeToDisplayFor : (NSString *)key resolver : ( + RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { resolve([RNSentryTimeToDisplay popTimeToDisplayFor:key]); } @@ -803,10 +982,8 @@ + (SentryUser *_Nullable)userFrom:(NSDictionary *)userKeys return @YES; // The return ensures that the method is synchronous } -RCT_EXPORT_METHOD(encodeToBase64 - : (NSArray *)array resolver - : (RCTPromiseResolveBlock)resolve rejecter - : (RCTPromiseRejectBlock)reject) +RCT_EXPORT_METHOD(encodeToBase64 : (NSArray *)array resolver : ( + RCTPromiseResolveBlock)resolve rejecter : (RCTPromiseRejectBlock)reject) { NSUInteger count = array.count; uint8_t *bytes = (uint8_t *)malloc(count); diff --git a/packages/core/ios/RNSentryExperimentalOptions.h b/packages/core/ios/RNSentryExperimentalOptions.h new file mode 100644 index 0000000000..ec0501cb05 --- /dev/null +++ b/packages/core/ios/RNSentryExperimentalOptions.h @@ -0,0 +1,42 @@ +#import + +@class SentryOptions; + +NS_ASSUME_NONNULL_BEGIN + +@interface RNSentryExperimentalOptions : NSObject + +/** + * Sets the enableUnhandledCPPExceptionsV2 experimental option on SentryOptions + * @param sentryOptions The SentryOptions instance to configure + * @param enabled Whether to enable unhandled C++ exceptions V2 + */ ++ (void)setEnableUnhandledCPPExceptionsV2:(BOOL)enabled + sentryOptions:(SentryOptions *)sentryOptions; + +/** + * Gets the current value of enableUnhandledCPPExceptionsV2 experimental option + * @param sentryOptions The SentryOptions instance to read from + * @return The current value of enableUnhandledCPPExceptionsV2 + */ ++ (BOOL)getEnableUnhandledCPPExceptionsV2:(SentryOptions *)sentryOptions; + +/** + * Sets the enableLogs experimental option on SentryOptions + * @param sentryOptions The SentryOptions instance to configure + * @param enabled Whether logs from sentry Cocoa should be enabled + */ ++ (void)setEnableLogs:(BOOL)enabled sentryOptions:(SentryOptions *)sentryOptions; + +/** + * Sets the enableSessionReplayInUnreliableEnvironment experimental option on SentryOptions + * @param sentryOptions The SentryOptions instance to configure + * @param enabled Whether enableSessionReplayInUnreliableEnvironment from sentry Cocoa should be + * enabled + */ ++ (void)setEnableSessionReplayInUnreliableEnvironment:(BOOL)enabled + sentryOptions:(SentryOptions *)sentryOptions; + +@end + +NS_ASSUME_NONNULL_END diff --git a/packages/core/ios/RNSentryExperimentalOptions.m b/packages/core/ios/RNSentryExperimentalOptions.m new file mode 100644 index 0000000000..7e0974e527 --- /dev/null +++ b/packages/core/ios/RNSentryExperimentalOptions.m @@ -0,0 +1,39 @@ +#import "RNSentryExperimentalOptions.h" +@import Sentry; + +@implementation RNSentryExperimentalOptions + ++ (void)setEnableUnhandledCPPExceptionsV2:(BOOL)enabled sentryOptions:(SentryOptions *)sentryOptions +{ + if (sentryOptions == nil) { + return; + } + sentryOptions.experimental.enableUnhandledCPPExceptionsV2 = enabled; +} + ++ (BOOL)getEnableUnhandledCPPExceptionsV2:(SentryOptions *)sentryOptions +{ + if (sentryOptions == nil) { + return NO; + } + return sentryOptions.experimental.enableUnhandledCPPExceptionsV2; +} + ++ (void)setEnableLogs:(BOOL)enabled sentryOptions:(SentryOptions *)sentryOptions +{ + if (sentryOptions == nil) { + return; + } + sentryOptions.experimental.enableLogs = enabled; +} + ++ (void)setEnableSessionReplayInUnreliableEnvironment:(BOOL)enabled + sentryOptions:(SentryOptions *)sentryOptions +{ + if (sentryOptions == nil) { + return; + } + sentryOptions.experimental.enableSessionReplayInUnreliableEnvironment = enabled; +} + +@end diff --git a/packages/core/ios/RNSentryReplay.h b/packages/core/ios/RNSentryReplay.h index 452914af15..cda7035550 100644 --- a/packages/core/ios/RNSentryReplay.h +++ b/packages/core/ios/RNSentryReplay.h @@ -1,7 +1,11 @@ @interface RNSentryReplay : NSObject -+ (void)updateOptions:(NSMutableDictionary *)options; +/** + * Updates the session replay options + * @return true when session replay is enabled + */ ++ (BOOL)updateOptions:(NSMutableDictionary *)options; + (void)postInit; diff --git a/packages/core/ios/RNSentryReplay.mm b/packages/core/ios/RNSentryReplay.mm index 994ec36189..94fa30b4e4 100644 --- a/packages/core/ios/RNSentryReplay.mm +++ b/packages/core/ios/RNSentryReplay.mm @@ -1,5 +1,6 @@ #import "RNSentryReplay.h" #import "RNSentryReplayBreadcrumbConverterHelper.h" +#import "RNSentryReplayQuality.h" #import "RNSentryVersion.h" #import "React/RCTTextView.h" #import "Replay/RNSentryReplayMask.h" @@ -11,20 +12,25 @@ @implementation RNSentryReplay { } -+ (void)updateOptions:(NSMutableDictionary *)options ++ (BOOL)updateOptions:(NSMutableDictionary *)options { - if (options[@"replaysSessionSampleRate"] == nil - && options[@"replaysOnErrorSampleRate"] == nil) { + NSNumber *sessionSampleRate = options[@"replaysSessionSampleRate"]; + NSNumber *errorSampleRate = options[@"replaysOnErrorSampleRate"]; + + if (sessionSampleRate == nil && errorSampleRate == nil) { NSLog(@"Session replay disabled via configuration"); - return; + return NO; } NSLog(@"Setting up session replay"); NSDictionary *replayOptions = options[@"mobileReplayOptions"] ?: @{}; + NSString *qualityString = options[@"replaysSessionQuality"]; + [options setValue:@{ - @"sessionSampleRate" : options[@"replaysSessionSampleRate"] ?: [NSNull null], - @"errorSampleRate" : options[@"replaysOnErrorSampleRate"] ?: [NSNull null], + @"sessionSampleRate" : sessionSampleRate ?: [NSNull null], + @"errorSampleRate" : errorSampleRate ?: [NSNull null], + @"quality" : @([RNSentryReplayQuality parseReplayQuality:qualityString]), @"maskAllImages" : replayOptions[@"maskAllImages"] ?: [NSNull null], @"maskAllText" : replayOptions[@"maskAllText"] ?: [NSNull null], @"enableViewRendererV2" : replayOptions[@"enableViewRendererV2"] ?: [NSNull null], @@ -34,6 +40,8 @@ + (void)updateOptions:(NSMutableDictionary *)options @ { @"name" : REACT_NATIVE_SDK_NAME, @"version" : REACT_NATIVE_SDK_PACKAGE_VERSION } } forKey:@"sessionReplay"]; + return (errorSampleRate != nil && [errorSampleRate doubleValue] > 0) + || (sessionSampleRate != nil && [sessionSampleRate doubleValue] > 0); } + (NSArray *_Nonnull)getReplayRNRedactClasses:(NSDictionary *_Nullable)replayOptions diff --git a/packages/core/ios/RNSentryReplayBreadcrumbConverter.h b/packages/core/ios/RNSentryReplayBreadcrumbConverter.h index 2eee190e26..1a36919a29 100644 --- a/packages/core/ios/RNSentryReplayBreadcrumbConverter.h +++ b/packages/core/ios/RNSentryReplayBreadcrumbConverter.h @@ -1,7 +1,7 @@ @import Sentry; #if SENTRY_TARGET_REPLAY_SUPPORTED -@class SentryRRWebEvent; +@protocol SentryRRWebEvent; @interface RNSentryReplayBreadcrumbConverter : NSObject diff --git a/packages/core/ios/RNSentryReplayQuality.h b/packages/core/ios/RNSentryReplayQuality.h new file mode 100644 index 0000000000..290c149d78 --- /dev/null +++ b/packages/core/ios/RNSentryReplayQuality.h @@ -0,0 +1,13 @@ +#import + +NS_ASSUME_NONNULL_BEGIN + +typedef NS_ENUM(NSInteger, SentryReplayQuality); + +@interface RNSentryReplayQuality : NSObject + ++ (SentryReplayQuality)parseReplayQuality:(NSString *_Nullable)qualityString; + +@end + +NS_ASSUME_NONNULL_END diff --git a/packages/core/ios/RNSentryReplayQuality.m b/packages/core/ios/RNSentryReplayQuality.m new file mode 100644 index 0000000000..8982cb93e3 --- /dev/null +++ b/packages/core/ios/RNSentryReplayQuality.m @@ -0,0 +1,25 @@ +#import "RNSentryReplayQuality.h" +@import Sentry; + +@implementation RNSentryReplayQuality + ++ (SentryReplayQuality)parseReplayQuality:(NSString *_Nullable)qualityString +{ + if (qualityString == nil) { + return SentryReplayQualityMedium; + } + + NSString *lowercaseQuality = [qualityString lowercaseString]; + + if ([lowercaseQuality isEqualToString:@"low"]) { + return SentryReplayQualityLow; + } else if ([lowercaseQuality isEqualToString:@"medium"]) { + return SentryReplayQualityMedium; + } else if ([lowercaseQuality isEqualToString:@"high"]) { + return SentryReplayQualityHigh; + } else { + return SentryReplayQualityMedium; + } +} + +@end diff --git a/packages/core/ios/RNSentryStart.m b/packages/core/ios/RNSentryStart.m index 84e2d83b02..b94dd42c23 100644 --- a/packages/core/ios/RNSentryStart.m +++ b/packages/core/ios/RNSentryStart.m @@ -99,6 +99,72 @@ + (SentryOptions *_Nullable)createOptionsWithDictionary:(NSDictionary *_Nonnull) return sentryOptions; } +- (void)trySetIgnoreErrors:(NSMutableDictionary *)options +{ + NSArray *ignoreErrorsStr = nil; + NSArray *ignoreErrorsRegex = nil; + + id strArr = [options objectForKey:@"ignoreErrorsStr"]; + id regexArr = [options objectForKey:@"ignoreErrorsRegex"]; + if ([strArr isKindOfClass:[NSArray class]]) { + ignoreErrorsStr = (NSArray *)strArr; + } + if ([regexArr isKindOfClass:[NSArray class]]) { + ignoreErrorsRegex = (NSArray *)regexArr; + } + + NSMutableArray *strs = [NSMutableArray array]; + NSMutableArray *regexes = [NSMutableArray array]; + + if (ignoreErrorsStr != nil) { + for (id str in ignoreErrorsStr) { + if ([str isKindOfClass:[NSString class]]) { + [strs addObject:str]; + } + } + } + + if (ignoreErrorsRegex != nil) { + for (id pattern in ignoreErrorsRegex) { + if ([pattern isKindOfClass:[NSString class]]) { + NSError *error = nil; + NSRegularExpression *regex = + [NSRegularExpression regularExpressionWithPattern:pattern + options:0 + error:&error]; + if (regex && error == nil) { + [regexes addObject:regex]; + } + } + } + } + + _ignoreErrorPatternsStr = [strs count] > 0 ? [strs copy] : nil; + _ignoreErrorPatternsRegex = [regexes count] > 0 ? [regexes copy] : nil; +} + +- (BOOL)shouldIgnoreError:(NSString *)message +{ + if ((!_ignoreErrorPatternsStr && !_ignoreErrorPatternsRegex) || !message) { + return NO; + } + + for (NSString *str in _ignoreErrorPatternsStr) { + if ([message containsString:str]) { + return YES; + } + } + + for (NSRegularExpression *regex in _ignoreErrorPatternsRegex) { + NSRange range = NSMakeRange(0, message.length); + if ([regex firstMatchInString:message options:0 range:range]) { + return YES; + } + } + + return NO; +} + /** * This function updates the options with RNSentry defaults. These default can be * overwritten by users during manual native initialization. @@ -130,6 +196,18 @@ + (void)updateWithReactFinals:(SentryOptions *)options != NSNotFound) { return nil; } + + // Regex and Str are set when one of them has value so we only need to check one of them. + if (self->_ignoreErrorPatternsStr || self->_ignoreErrorPatternsRegex) { + for (SentryException *exception in event.exceptions) { + if ([self shouldIgnoreError:exception.value]) { + return nil; + } + } + if ([self shouldIgnoreError:event.message.message]) { + return nil; + } + } [self setEventOriginTag:event]; if (userBeforeSend == nil) { diff --git a/packages/core/ios/RNSentryVersion.m b/packages/core/ios/RNSentryVersion.m index d868d3b7a6..7849fd1fa0 100644 --- a/packages/core/ios/RNSentryVersion.m +++ b/packages/core/ios/RNSentryVersion.m @@ -3,4 +3,4 @@ NSString *const NATIVE_SDK_NAME = @"sentry.cocoa.react-native"; NSString *const REACT_NATIVE_SDK_NAME = @"sentry.javascript.react-native"; NSString *const REACT_NATIVE_SDK_PACKAGE_NAME = @"npm:@sentry/react-native"; -NSString *const REACT_NATIVE_SDK_PACKAGE_VERSION = @"6.15.1"; +NSString *const REACT_NATIVE_SDK_PACKAGE_VERSION = @"7.4.0"; diff --git a/packages/core/ios/Replay/RNSentryReplayMask.mm b/packages/core/ios/Replay/RNSentryReplayMask.mm index bc39f229e2..14453e26af 100644 --- a/packages/core/ios/Replay/RNSentryReplayMask.mm +++ b/packages/core/ios/Replay/RNSentryReplayMask.mm @@ -23,8 +23,7 @@ - (UIView *)view @end # ifdef RCT_NEW_ARCH_ENABLED -@interface -RNSentryReplayMask () +@interface RNSentryReplayMask () @end # endif diff --git a/packages/core/ios/Replay/RNSentryReplayUnmask.mm b/packages/core/ios/Replay/RNSentryReplayUnmask.mm index 8dd0f06611..f0ec5139d6 100644 --- a/packages/core/ios/Replay/RNSentryReplayUnmask.mm +++ b/packages/core/ios/Replay/RNSentryReplayUnmask.mm @@ -23,8 +23,7 @@ - (UIView *)view @end # ifdef RCT_NEW_ARCH_ENABLED -@interface -RNSentryReplayUnmask () +@interface RNSentryReplayUnmask () @end # endif diff --git a/packages/core/ios/SentrySDKWrapper.h b/packages/core/ios/SentrySDKWrapper.h new file mode 100644 index 0000000000..9040d032ab --- /dev/null +++ b/packages/core/ios/SentrySDKWrapper.h @@ -0,0 +1,18 @@ +#import + +@class SentryOptions; +@class SentryScope; + +@interface SentrySDKWrapper : NSObject + ++ (void)configureScope:(void (^)(SentryScope *scope))callback; + ++ (void)crash; + ++ (void)close; + ++ (BOOL)crashedLastRun; + ++ (void)startWithOptions:(SentryOptions *)options; + +@end diff --git a/packages/core/ios/SentrySDKWrapper.m b/packages/core/ios/SentrySDKWrapper.m new file mode 100644 index 0000000000..0845f1ff3e --- /dev/null +++ b/packages/core/ios/SentrySDKWrapper.m @@ -0,0 +1,31 @@ +#import "SentrySDKWrapper.h" +@import Sentry; + +@implementation SentrySDKWrapper + ++ (void)startWithOptions:(SentryOptions *)options +{ + [SentrySDK startWithOptions:options]; +} + ++ (void)crash +{ + [SentrySDK crash]; +} + ++ (void)close +{ + [SentrySDK close]; +} + ++ (BOOL)crashedLastRun +{ + return [SentrySDK crashedLastRun]; +} + ++ (void)configureScope:(void (^)(SentryScope *scope))callback +{ + [SentrySDK configureScope:callback]; +} + +@end diff --git a/packages/core/package.json b/packages/core/package.json index 489784211d..4094e3d42f 100644 --- a/packages/core/package.json +++ b/packages/core/package.json @@ -2,7 +2,7 @@ "name": "@sentry/react-native", "homepage": "https://github.com/getsentry/sentry-react-native", "repository": "https://github.com/getsentry/sentry-react-native", - "version": "6.15.1", + "version": "7.4.0", "description": "Official Sentry SDK for react-native", "typings": "dist/js/index.d.ts", "types": "dist/js/index.d.ts", @@ -18,19 +18,22 @@ }, "main": "dist/js/index.js", "scripts": { - "build": "npx run-s build:sdk downlevel build:tools build:plugin", + "build": "npx run-s build:sdk downlevel build:tools build:plugin build:replay-stubs", "build:sdk": "tsc -p tsconfig.build.json", "build:sdk:watch": "tsc -p tsconfig.build.json -w --preserveWatchOutput", "build:tools": "tsc -p tsconfig.build.tools.json", "build:tools:watch": "tsc -p tsconfig.build.tools.json -w --preserveWatchOutput", "build:plugin": "EXPO_NONINTERACTIVE=true expo-module build plugin", + "build:replay-stubs": "cd android/replay-stubs && ./gradlew jar", "build:tarball": "bash scripts/build-tarball.sh", "downlevel": "downlevel-dts dist ts3.8/dist --to=3.8", "clean": "rimraf dist coverage && yarn clean:plugin", "clean:plugin": "expo-module clean plugin", "circularDepCheck": "madge --circular dist/js/index.js && madge --circular metro.js && madge --circular expo.js", "test": "yarn test:sdk && yarn test:tools", - "test:sdk": "npx jest", + "test:sdk": "sh -c 'if [ \"$CI\" = \"true\" ]; then yarn test:sdk-ci; else yarn test:sdk-local; fi'", + "test:sdk-ci": "npx jest", + "test:sdk-local": "jest --maxWorkers=8", "test:tools": "npx jest --config jest.config.tools.js", "test:watch": "npx jest --watch", "yalc:add:sentry-javascript": "yalc add @sentry/browser @sentry/core @sentry/react @sentry/types", @@ -65,23 +68,22 @@ "react-native": ">=0.65.0" }, "dependencies": { - "@sentry/babel-plugin-component-annotate": "3.5.0", - "@sentry/browser": "8.54.0", - "@sentry/cli": "2.46.0", - "@sentry/core": "8.54.0", - "@sentry/react": "8.54.0", - "@sentry/types": "8.54.0", - "@sentry/utils": "8.54.0" + "@sentry/babel-plugin-component-annotate": "4.4.0", + "@sentry/browser": "10.20.0", + "@sentry/cli": "2.56.1", + "@sentry/core": "10.20.0", + "@sentry/react": "10.20.0", + "@sentry/types": "10.20.0" }, "devDependencies": { "@babel/core": "^7.25.2", "@expo/metro-config": "~0.20.0", "@mswjs/interceptors": "^0.25.15", "@react-native/babel-preset": "0.77.1", - "@sentry-internal/eslint-config-sdk": "8.54.0", - "@sentry-internal/eslint-plugin-sdk": "8.54.0", - "@sentry-internal/typescript": "8.54.0", - "@sentry/wizard": "5.1.0", + "@sentry-internal/eslint-config-sdk": "10.20.0", + "@sentry-internal/eslint-plugin-sdk": "10.20.0", + "@sentry-internal/typescript": "10.20.0", + "@sentry/wizard": "6.6.0", "@testing-library/react-native": "^12.7.2", "@types/jest": "^29.5.13", "@types/node": "^20.9.3", @@ -104,13 +106,12 @@ "jest-environment-jsdom": "^29.6.2", "jest-extended": "^4.0.2", "madge": "^6.1.0", - "metro": "0.81.0", + "metro": "0.83.1", "prettier": "^2.0.5", "react": "18.3.1", "react-native": "0.77.1", - "react-test-renderer": "^18.3.1", "rimraf": "^4.1.1", - "ts-jest": "^29.1.1", + "ts-jest": "^29.3.1", "typescript": "4.9.5", "uglify-js": "^3.17.4", "uuid": "^9.0.1", diff --git a/packages/core/playground.d.ts b/packages/core/playground.d.ts new file mode 100644 index 0000000000..6f0a924c28 --- /dev/null +++ b/packages/core/playground.d.ts @@ -0,0 +1 @@ +export * from './dist/js/playground'; diff --git a/packages/core/playground.js b/packages/core/playground.js new file mode 100644 index 0000000000..6f0a924c28 --- /dev/null +++ b/packages/core/playground.js @@ -0,0 +1 @@ +export * from './dist/js/playground'; diff --git a/packages/core/playground_animations/bug.gif b/packages/core/playground_animations/bug.gif new file mode 100644 index 0000000000..9b8223b7db Binary files /dev/null and b/packages/core/playground_animations/bug.gif differ diff --git a/packages/core/playground_animations/hi.gif b/packages/core/playground_animations/hi.gif new file mode 100644 index 0000000000..19d0bedebf Binary files /dev/null and b/packages/core/playground_animations/hi.gif differ diff --git a/packages/core/playground_animations/thumbsup.gif b/packages/core/playground_animations/thumbsup.gif new file mode 100644 index 0000000000..948f4e292e Binary files /dev/null and b/packages/core/playground_animations/thumbsup.gif differ diff --git a/packages/core/plugin/src/withSentry.ts b/packages/core/plugin/src/withSentry.ts index 70d4c8932b..2fdee7f063 100644 --- a/packages/core/plugin/src/withSentry.ts +++ b/packages/core/plugin/src/withSentry.ts @@ -1,6 +1,5 @@ import type { ConfigPlugin } from 'expo/config-plugins'; import { createRunOncePlugin } from 'expo/config-plugins'; - import { bold, sdkPackage, warnOnce } from './utils'; import { withSentryAndroid } from './withSentryAndroid'; import type { SentryAndroidGradlePluginOptions } from './withSentryAndroidGradlePlugin'; @@ -18,7 +17,7 @@ interface PluginProps { const withSentryPlugin: ConfigPlugin = (config, props) => { const sentryProperties = getSentryProperties(props); - if (props && props.authToken) { + if (props?.authToken) { // If not removed, the plugin config with the authToken will be written to the application package delete props.authToken; } @@ -50,8 +49,9 @@ const withSentryPlugin: ConfigPlugin = (config, props) => { const missingProjectMessage = '# no project found, falling back to SENTRY_PROJECT environment variable'; const missingOrgMessage = '# no org found, falling back to SENTRY_ORG environment variable'; -const existingAuthTokenMessage = `# DO NOT COMMIT the auth token, use SENTRY_AUTH_TOKEN instead, see https://docs.sentry.io/platforms/react-native/manual-setup/`; -const missingAuthTokenMessage = `# Using SENTRY_AUTH_TOKEN environment variable`; +const existingAuthTokenMessage = + '# DO NOT COMMIT the auth token, use SENTRY_AUTH_TOKEN instead, see https://docs.sentry.io/platforms/react-native/manual-setup/'; +const missingAuthTokenMessage = '# Using SENTRY_AUTH_TOKEN environment variable'; export function getSentryProperties(props: PluginProps | void): string | null { const { organization, project, authToken, url = 'https://sentry.io/' } = props ?? {}; diff --git a/packages/core/plugin/src/withSentryAndroid.ts b/packages/core/plugin/src/withSentryAndroid.ts index 9beaa23883..83c92b464b 100644 --- a/packages/core/plugin/src/withSentryAndroid.ts +++ b/packages/core/plugin/src/withSentryAndroid.ts @@ -1,23 +1,22 @@ import type { ConfigPlugin } from 'expo/config-plugins'; import { withAppBuildGradle, withDangerousMod } from 'expo/config-plugins'; import * as path from 'path'; - import { warnOnce, writeSentryPropertiesTo } from './utils'; export const withSentryAndroid: ConfigPlugin = (config, sentryProperties: string) => { - const cfg = withAppBuildGradle(config, config => { - if (config.modResults.language === 'groovy') { - config.modResults.contents = modifyAppBuildGradle(config.modResults.contents); + const cfg = withAppBuildGradle(config, appBuildGradle => { + if (appBuildGradle.modResults.language === 'groovy') { + appBuildGradle.modResults.contents = modifyAppBuildGradle(appBuildGradle.modResults.contents); } else { throw new Error('Cannot configure Sentry in the app gradle because the build.gradle is not groovy'); } - return config; + return appBuildGradle; }); return withDangerousMod(cfg, [ 'android', - config => { - writeSentryPropertiesTo(path.resolve(config.modRequest.projectRoot, 'android'), sentryProperties); - return config; + dangerousMod => { + writeSentryPropertiesTo(path.resolve(dangerousMod.modRequest.projectRoot, 'android'), sentryProperties); + return dangerousMod; }, ]); }; diff --git a/packages/core/plugin/src/withSentryAndroidGradlePlugin.ts b/packages/core/plugin/src/withSentryAndroidGradlePlugin.ts index 27a9a4d904..9b35052536 100644 --- a/packages/core/plugin/src/withSentryAndroidGradlePlugin.ts +++ b/packages/core/plugin/src/withSentryAndroidGradlePlugin.ts @@ -1,5 +1,5 @@ import { withAppBuildGradle, withProjectBuildGradle } from '@expo/config-plugins'; - +import type { ExpoConfig } from '@expo/config-types'; import { warnOnce } from './utils'; export interface SentryAndroidGradlePluginOptions { @@ -13,12 +13,14 @@ export interface SentryAndroidGradlePluginOptions { includeSourceContext?: boolean; } +export const sentryAndroidGradlePluginVersion = '5.12.1'; + /** * Adds the Sentry Android Gradle Plugin to the project. * https://docs.sentry.io/platforms/react-native/manual-setup/manual-setup/#enable-sentry-agp */ export function withSentryAndroidGradlePlugin( - config: any, + config: ExpoConfig, { includeProguardMapping = true, dexguardEnabled = false, @@ -28,60 +30,51 @@ export function withSentryAndroidGradlePlugin( includeNativeSources = true, includeSourceContext = false, }: SentryAndroidGradlePluginOptions = {}, -): any { - const version = '4.14.1'; - +): ExpoConfig { // Modify android/build.gradle - const withSentryProjectBuildGradle = (config: any): any => { - return withProjectBuildGradle(config, (projectBuildGradle: any) => { - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access - if (!projectBuildGradle.modResults || !projectBuildGradle.modResults.contents) { + const withSentryProjectBuildGradle = (config: ExpoConfig): ExpoConfig => { + return withProjectBuildGradle(config, projectBuildGradle => { + if (!projectBuildGradle.modResults?.contents) { warnOnce('android/build.gradle content is missing or undefined.'); - return config; + return projectBuildGradle; } - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access if (projectBuildGradle.modResults.language !== 'groovy') { warnOnce('Cannot configure Sentry in android/build.gradle because it is not in Groovy.'); - return config; + return projectBuildGradle; } - const dependency = `classpath("io.sentry:sentry-android-gradle-plugin:${version}")`; + const dependency = `classpath("io.sentry:sentry-android-gradle-plugin:${sentryAndroidGradlePluginVersion}")`; - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access if (projectBuildGradle.modResults.contents.includes(dependency)) { warnOnce('sentry-android-gradle-plugin dependency in already in android/build.gradle.'); - return config; + return projectBuildGradle; } try { - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access const updatedContents = projectBuildGradle.modResults.contents.replace( /dependencies\s*{/, `dependencies {\n ${dependency}`, ); - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access if (updatedContents === projectBuildGradle.modResults.contents) { warnOnce('Failed to inject the dependency. Could not find `dependencies` in build.gradle.'); } else { - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access projectBuildGradle.modResults.contents = updatedContents; } } catch (error) { - warnOnce(`An error occurred while trying to modify build.gradle`); + warnOnce('An error occurred while trying to modify build.gradle'); } return projectBuildGradle; }); }; // Modify android/app/build.gradle - const withSentryAppBuildGradle = (config: any): any => { - return withAppBuildGradle(config, (config: any) => { - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access - if (config.modResults.language !== 'groovy') { + const withSentryAppBuildGradle = (config: ExpoConfig): ExpoConfig => { + return withAppBuildGradle(config, appBuildGradle => { + if (appBuildGradle.modResults.language !== 'groovy') { warnOnce('Cannot configure Sentry in android/app/build.gradle because it is not in Groovy.'); - return config; + return appBuildGradle; } - const sentryPlugin = `apply plugin: "io.sentry.android.gradle"`; + const sentryPlugin = 'apply plugin: "io.sentry.android.gradle"'; const sentryConfig = ` sentry { autoUploadProguardMapping = ${autoUploadProguardMapping ? 'shouldSentryAutoUpload()' : 'false'} @@ -99,22 +92,18 @@ export function withSentryAndroidGradlePlugin( } }`; - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access - let contents = config.modResults.contents; + let contents = appBuildGradle.modResults.contents; - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access if (!contents.includes(sentryPlugin)) { contents = `${sentryPlugin}\n${contents}`; } - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access if (!contents.includes('sentry {')) { contents = `${contents}\n${sentryConfig}`; } - // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access - config.modResults.contents = contents; - return config; + appBuildGradle.modResults.contents = contents; + return appBuildGradle; }); }; diff --git a/packages/core/plugin/src/withSentryIOS.ts b/packages/core/plugin/src/withSentryIOS.ts index db25261839..e10f820282 100644 --- a/packages/core/plugin/src/withSentryIOS.ts +++ b/packages/core/plugin/src/withSentryIOS.ts @@ -2,7 +2,6 @@ import type { ConfigPlugin, XcodeProject } from 'expo/config-plugins'; import { withDangerousMod, withXcodeProject } from 'expo/config-plugins'; import * as path from 'path'; - import { warnOnce, writeSentryPropertiesTo } from './utils'; type BuildPhase = { shellScript: string }; diff --git a/packages/core/react-native.config.js b/packages/core/react-native.config.js index d307073e1a..5281f24094 100644 --- a/packages/core/react-native.config.js +++ b/packages/core/react-native.config.js @@ -4,8 +4,8 @@ module.exports = { ios: {}, android: { packageInstance: 'new RNSentryPackage()', - packageImportPath: 'import io.sentry.react.RNSentryPackage;' - } - } - } + packageImportPath: 'import io.sentry.react.RNSentryPackage;', + }, + }, + }, }; diff --git a/packages/core/scripts/expo-upload-sourcemaps.js b/packages/core/scripts/expo-upload-sourcemaps.js index c282adeb3f..d33afc95f6 100755 --- a/packages/core/scripts/expo-upload-sourcemaps.js +++ b/packages/core/scripts/expo-upload-sourcemaps.js @@ -129,7 +129,10 @@ try { console.warn(error); } -loadDotenv(path.join(projectRoot, '.env.sentry-build-plugin')); +const sentryBuildPluginPath = path.join(projectRoot, '.env.sentry-build-plugin'); +if (fs.existsSync(sentryBuildPluginPath)) { + loadDotenv(sentryBuildPluginPath); +} let sentryOrg = getEnvVar(SENTRY_ORG); let sentryUrl = getEnvVar(SENTRY_URL); @@ -174,7 +177,7 @@ if (!sentryOrg || !sentryProject || !sentryUrl) { console.log(`${SENTRY_URL} resolved to ${sentryUrl} from expo config.`); } else { - sentryUrl = `https://sentry.io/`; + sentryUrl = 'https://sentry.io/'; console.log( `Since it wasn't specified in the Expo config or environment variable, ${SENTRY_URL} now points to ${sentryUrl}.` ); @@ -214,7 +217,7 @@ for (const [assetGroupName, assets] of Object.entries(groupedAssets)) { } const isHermes = assets.find(asset => asset.endsWith('.hbc')); - const windowsCallback = process.platform === "win32" ? 'node ' : ''; + const windowsCallback = process.platform === 'win32' ? 'node ' : ''; execSync(`${windowsCallback}${sentryCliBin} sourcemaps upload ${isHermes ? '--debug-id-reference' : ''} ${assets.join(' ')}`, { env: { ...process.env, @@ -231,7 +234,7 @@ if (numAssetsUploaded === totalAssets) { console.log('✅ Uploaded bundles and sourcemaps to Sentry successfully.'); } else { console.warn( - `⚠️ Uploaded ${numAssetsUploaded} of ${totalAssets} bundles and sourcemaps. ${numAssetsUploaded === 0 ? 'Ensure you are running `expo export` with the `--dump-sourcemap` flag.' : '' + `⚠️ Uploaded ${numAssetsUploaded} of ${totalAssets} bundles and sourcemaps. ${numAssetsUploaded === 0 ? 'Ensure you are running `expo export` with the `--source-maps` flag.' : '' }`, ); } diff --git a/packages/core/scripts/sentry-xcode-debug-files.sh b/packages/core/scripts/sentry-xcode-debug-files.sh index 29e8e3967c..9ebf0ae2bc 100755 --- a/packages/core/scripts/sentry-xcode-debug-files.sh +++ b/packages/core/scripts/sentry-xcode-debug-files.sh @@ -24,10 +24,33 @@ LOCAL_NODE_BINARY=${NODE_BINARY:-node} RN_PROJECT_ROOT="${PROJECT_DIR}/.." [ -z "$SENTRY_PROPERTIES" ] && export SENTRY_PROPERTIES=sentry.properties -[ -z "$SENTRY_DOTENV_PATH" ] && export SENTRY_DOTENV_PATH="$RN_PROJECT_ROOT/.env.sentry-build-plugin" +[ -z "$SENTRY_DOTENV_PATH" ] && [ -f "$RN_PROJECT_ROOT/.env.sentry-build-plugin" ] && export SENTRY_DOTENV_PATH="$RN_PROJECT_ROOT/.env.sentry-build-plugin" [ -z "$SENTRY_CLI_EXECUTABLE" ] && SENTRY_CLI_PACKAGE_PATH=$("$LOCAL_NODE_BINARY" --print "require('path').dirname(require.resolve('@sentry/cli/package.json'))") -[ -z "$SENTRY_CLI_EXECUTABLE" ] && SENTRY_CLI_EXECUTABLE="${SENTRY_CLI_PACKAGE_PATH}/bin/sentry-cli" +[ -z "$SOURCEMAP_FILE" ] && export SOURCEMAP_FILE="$DERIVED_FILE_DIR/main.jsbundle.map" + +if [ -z "$SENTRY_CLI_EXECUTABLE" ]; then + # Try standard resolution safely + RESOLVED_PATH=$( + "$LOCAL_NODE_BINARY" --print "require('path').dirname(require.resolve('@sentry/cli/package.json'))" 2>/dev/null + ) || true + if [ -n "$RESOLVED_PATH" ]; then + SENTRY_CLI_PACKAGE_PATH="$RESOLVED_PATH/bin/sentry-cli" + else + # Fallback: parse NODE_PATH from the .bin/sentry-cli shim (file generated by PNPM) + PNPM_BIN_PATH="$PWD/../node_modules/@sentry/react-native/node_modules/.bin/sentry-cli" + + if [ -f "$PNPM_BIN_PATH" ]; then + CLI_FILE_TEXT=$(cat "$PNPM_BIN_PATH") + + # Filter where PNPM stored Sentry CLI + NODE_PATH_LINE=$(echo "$CLI_FILE_TEXT" | grep -oE 'NODE_PATH="[^"]+"' | head -n1) + NODE_PATH_VALUE=$(echo "$NODE_PATH_LINE" | sed -E 's/^NODE_PATH="([^"]+)".*/\1/') + SENTRY_CLI_PACKAGE_PATH=${NODE_PATH_VALUE%%/bin*} + fi + fi +fi +[ -z "$SENTRY_CLI_EXECUTABLE" ] && SENTRY_CLI_EXECUTABLE="$SENTRY_CLI_PACKAGE_PATH" [[ $SENTRY_INCLUDE_NATIVE_SOURCES == "true" ]] && INCLUDE_SOURCES_FLAG="--include-sources" || INCLUDE_SOURCES_FLAG="" diff --git a/packages/core/scripts/sentry-xcode.sh b/packages/core/scripts/sentry-xcode.sh index 6d0764b90a..c9593e566b 100755 --- a/packages/core/scripts/sentry-xcode.sh +++ b/packages/core/scripts/sentry-xcode.sh @@ -13,13 +13,34 @@ LOCAL_NODE_BINARY=${NODE_BINARY:-node} RN_PROJECT_ROOT="${PROJECT_DIR}/.." [ -z "$SENTRY_PROPERTIES" ] && export SENTRY_PROPERTIES=sentry.properties -[ -z "$SENTRY_DOTENV_PATH" ] && export SENTRY_DOTENV_PATH="$RN_PROJECT_ROOT/.env.sentry-build-plugin" +[ -z "$SENTRY_DOTENV_PATH" ] && [ -f "$RN_PROJECT_ROOT/.env.sentry-build-plugin" ] && export SENTRY_DOTENV_PATH="$RN_PROJECT_ROOT/.env.sentry-build-plugin" [ -z "$SOURCEMAP_FILE" ] && export SOURCEMAP_FILE="$DERIVED_FILE_DIR/main.jsbundle.map" -[ -z "$SENTRY_CLI_EXECUTABLE" ] && SENTRY_CLI_PACKAGE_PATH=$("$LOCAL_NODE_BINARY" --print "require('path').dirname(require.resolve('@sentry/cli/package.json'))") -[ -z "$SENTRY_CLI_EXECUTABLE" ] && SENTRY_CLI_EXECUTABLE="${SENTRY_CLI_PACKAGE_PATH}/bin/sentry-cli" +if [ -z "$SENTRY_CLI_EXECUTABLE" ]; then + # Try standard resolution safely + RESOLVED_PATH=$( + "$LOCAL_NODE_BINARY" --print "require('path').dirname(require.resolve('@sentry/cli/package.json'))" 2>/dev/null + ) || true + if [ -n "$RESOLVED_PATH" ]; then + SENTRY_CLI_PACKAGE_PATH="$RESOLVED_PATH/bin/sentry-cli" + else + # Fallback: parse NODE_PATH from the .bin/sentry-cli shim (file generated by PNPM) + PNPM_BIN_PATH="$PWD/../node_modules/@sentry/react-native/node_modules/.bin/sentry-cli" + + if [ -f "$PNPM_BIN_PATH" ]; then + CLI_FILE_TEXT=$(cat "$PNPM_BIN_PATH") + + # Filter where PNPM stored Sentry CLI + NODE_PATH_LINE=$(echo "$CLI_FILE_TEXT" | grep -oE 'NODE_PATH="[^"]+"' | head -n1) + NODE_PATH_VALUE=$(echo "$NODE_PATH_LINE" | sed -E 's/^NODE_PATH="([^"]+)".*/\1/') + SENTRY_CLI_PACKAGE_PATH=${NODE_PATH_VALUE%%/bin*} + fi + fi +fi +[ -z "$SENTRY_CLI_EXECUTABLE" ] && SENTRY_CLI_EXECUTABLE="$SENTRY_CLI_PACKAGE_PATH" -REACT_NATIVE_XCODE=$1 +REACT_NATIVE_XCODE_DEFAULT="../node_modules/react-native/scripts/react-native-xcode.sh" +REACT_NATIVE_XCODE="${1:-$REACT_NATIVE_XCODE_DEFAULT}" [[ "$AUTO_RELEASE" == false ]] && [[ -z "$BUNDLE_COMMAND" || "$BUNDLE_COMMAND" != "ram-bundle" ]] && NO_AUTO_RELEASE="--no-auto-release" ARGS="$NO_AUTO_RELEASE $SENTRY_CLI_EXTRA_ARGS $SENTRY_CLI_RN_XCODE_EXTRA_ARGS" diff --git a/packages/core/scripts/sentry_utils.rb b/packages/core/scripts/sentry_utils.rb index c5a8158c4d..5dc57a3b52 100644 --- a/packages/core/scripts/sentry_utils.rb +++ b/packages/core/scripts/sentry_utils.rb @@ -31,3 +31,12 @@ def is_hermes_default(rn_version) def is_profiling_supported(rn_version) return (rn_version[:major] >= 1 || (rn_version[:major] == 0 && rn_version[:minor] >= 69)) end + +# Check if we need the old Folly flags (for RN < 0.80.0) +def should_use_folly_flags(rn_version) + return (rn_version[:major] == 0 && rn_version[:minor] < 80) +end + +def is_new_hermes_runtime(rn_version) + return (rn_version[:major] >= 1 || (rn_version[:major] == 0 && rn_version[:minor] >= 81)) +end diff --git a/packages/core/sentry.gradle b/packages/core/sentry.gradle index 66f020d7bc..15d984b00f 100644 --- a/packages/core/sentry.gradle +++ b/packages/core/sentry.gradle @@ -19,6 +19,11 @@ project.ext.shouldCopySentryOptionsFile = { -> // If not set, default to true return System.getenv('SENTRY_COPY_OPTIONS_FILE') != 'false' } +interface InjectedExecOps { + @Inject //@javax.inject.Inject + ExecOperations getExecOps() +} + def config = project.hasProperty("sentryCli") ? project.sentryCli : []; def configFile = "sentry.options.json" // Sentry configuration file @@ -57,6 +62,259 @@ tasks.register("cleanupTemporarySentryJsonConfiguration") { } } +plugins.withId('com.android.application') { + def androidComponents = extensions.getByName("androidComponents") + + androidComponents.onVariants(androidComponents.selector().all()) { v -> + if (!v.name.toLowerCase().contains("debug")) { + // separately we then hook into the bundle task of react native to inject + // sourcemap generation parameters. In case for whatever reason no release + // was found for the asset folder we just bail. + def bundleTasks = tasks.findAll { task -> (task.name.startsWith("createBundle") || task.name.startsWith("bundle")) && task.name.endsWith("JsAndAssets") && !task.name.contains("Debug") && task.enabled } + bundleTasks.each { bundleTask -> + def shouldCleanUp + def sourcemapOutput + def bundleOutput + def packagerSourcemapOutput + def bundleCommand + def props = bundleTask.getProperties() + def reactRoot = props.get("workingDir") + if (reactRoot == null) { + reactRoot = props.get("root").get() // RN 0.71 and above + } + def modulesOutput = "$reactRoot/android/app/src/main/assets/modules.json" + def modulesTask = null + + (shouldCleanUp, bundleOutput, sourcemapOutput, packagerSourcemapOutput, bundleCommand) = forceSourceMapOutputFromBundleTask(bundleTask) + + // Lets leave this here if we need to debug + // println bundleTask.properties + // .sort{it.key} + // .collect{it} + // .findAll{!['class', 'active'].contains(it.key)} + // .join('\n') + + def currentVariants = extractCurrentVariants(bundleTask, v) + if (currentVariants == null) return + + def previousCliTask = null + def applicationVariant = null + + def nameCleanup = "${bundleTask.name}_SentryUploadCleanUp" + def nameModulesCleanup = "${bundleTask.name}_SentryCollectModulesCleanUp" + // Upload the source map several times if necessary: once for each release and versionCode. + currentVariants.each { key, currentVariant -> + def variant = currentVariant[0] + def releaseName = currentVariant[1] + def versionCode = currentVariant[2] + applicationVariant = currentVariant[3] + + try { + if (versionCode instanceof String) { + versionCode = Integer.parseInt(versionCode) + versionCode = Math.abs(versionCode) + } + } catch (NumberFormatException e) { + project.logger.info("versionCode: '$versionCode' isn't an Integer, using the plain value.") + } + + // The Sentry server distinguishes source maps by release (`--release` in the command + // below) and distribution identifier (`--dist` below). Give the task a unique name + // based on where we're uploading to. + def nameCliTask = "${bundleTask.name}_SentryUpload_${releaseName}_${versionCode}" + def nameModulesTask = "${bundleTask.name}_SentryCollectModules_${releaseName}_${versionCode}" + + // If several outputs have the same releaseName and versionCode, we'd do the exact same + // upload for each of them. No need to repeat. + try { tasks.named(nameCliTask); return } catch (Exception e) {} + + /** Upload source map file to the sentry server via CLI call. */ + def cliTask = tasks.register(nameCliTask) { + onlyIf { shouldSentryAutoUploadGeneral() } + description = "upload debug symbols to sentry" + group = 'sentry.io' + + def extraArgs = [] + + def sentryPackage = resolveSentryReactNativeSDKPath(reactRoot) + def copyDebugIdScript = config.copyDebugIdScript + ? file(config.copyDebugIdScript).getAbsolutePath() + : "$sentryPackage/scripts/copy-debugid.js" + def hasSourceMapDebugIdScript = config.hasSourceMapDebugIdScript + ? file(config.hasSourceMapDebugIdScript).getAbsolutePath() + : "$sentryPackage/scripts/has-sourcemap-debugid.js" + + def injected = project.objects.newInstance(InjectedExecOps) + doFirst { + // Copy Debug ID from packager source map to Hermes composed source map + injected.execOps.exec { + def args = ["node", + copyDebugIdScript, + packagerSourcemapOutput, + sourcemapOutput] + def osCompatibilityCopyCommand = Os.isFamily(Os.FAMILY_WINDOWS) ? ['cmd', '/c'] : [] + commandLine(*osCompatibilityCopyCommand, *args) + } + + // Add release and dist for backward compatibility if no Debug ID detected in output soruce map + def process = ["node", hasSourceMapDebugIdScript, sourcemapOutput].execute(null, new File("$reactRoot")) + def exitValue = process.waitFor() + project.logger.lifecycle("Check generated source map for Debug ID: ${process.text}") + + project.logger.lifecycle("Sentry Source Maps upload will include the release name and dist.") + extraArgs.addAll([ + "--release", releaseName, + "--dist", versionCode + ]) + } + + doLast { + injected.execOps.exec { + workingDir reactRoot + + def propertiesFile = config.sentryProperties + ? config.sentryProperties + : "$reactRoot/android/sentry.properties" + + if (config.flavorAware) { + propertiesFile = "$reactRoot/android/sentry-${variant}.properties" + project.logger.info("For $variant using: $propertiesFile") + } else { + environment("SENTRY_PROPERTIES", propertiesFile) + } + + Properties sentryProps = new Properties() + try { + sentryProps.load(new FileInputStream(propertiesFile)) + } catch (FileNotFoundException e) { + project.logger.info("file not found '$propertiesFile' for '$variant'") + } + + def cliPackage = resolveSentryCliPackagePath(reactRoot) + def cliExecutable = sentryProps.get("cli.executable", "$cliPackage/bin/sentry-cli") + + // fix path separator for Windows + if (Os.isFamily(Os.FAMILY_WINDOWS)) { + cliExecutable = cliExecutable.replaceAll("/", "\\\\") + } + + // + // based on: + // https://github.com/getsentry/sentry-cli/blob/master/src/commands/react_native_gradle.rs + // + def args = [cliExecutable] + + args.addAll(!config.logLevel ? [] : [ + "--log-level", config.logLevel // control verbosity of the output + ]) + args.addAll(!config.flavorAware ? [] : [ + "--url", sentryProps.get("defaults.url"), + "--auth-token", sentryProps.get("auth.token") ?: System.getenv("SENTRY_AUTH_TOKEN") + ]) + args.addAll(["react-native", "gradle", + "--bundle", bundleOutput, // The path to a bundle that should be uploaded. + "--sourcemap", sourcemapOutput // The path to a sourcemap that should be uploaded. + ]) + args.addAll(!config.flavorAware ? [] : [ + "--org", sentryProps.get("defaults.org"), + "--project", sentryProps.get("defaults.project") + ]) + + args.addAll(extraArgs) + + project.logger.lifecycle("Sentry-CLI arguments: ${args}") + def osCompatibility = Os.isFamily(Os.FAMILY_WINDOWS) ? ['cmd', '/c', 'node'] : [] + if (!System.getenv('SENTRY_DOTENV_PATH') && file("$reactRoot/.env.sentry-build-plugin").exists()) { + environment('SENTRY_DOTENV_PATH', "$reactRoot/.env.sentry-build-plugin") + } + commandLine(*osCompatibility, *args) + } + } + + enabled true + } + + modulesTask = tasks.register(nameModulesTask, Exec) { + description = "collect javascript modules from bundle source map" + group = 'sentry.io' + + workingDir reactRoot + + def sentryPackage = resolveSentryReactNativeSDKPath(reactRoot) + + def collectModulesScript = config.collectModulesScript + ? file(config.collectModulesScript).getAbsolutePath() + : "$sentryPackage/dist/js/tools/collectModules.js" + def modulesPaths = config.modulesPaths + ? config.modulesPaths.join(',') + : "$reactRoot/node_modules" + def args = ["node", + collectModulesScript, + sourcemapOutput, + modulesOutput, + modulesPaths + ] + + if ((new File(collectModulesScript)).exists()) { + project.logger.info("Sentry-CollectModules arguments: ${args}") + commandLine(*args) + + def skip = config.skipCollectModules + ? config.skipCollectModules == true + : false + enabled !skip + } else { + project.logger.info("collectModulesScript not found: $collectModulesScript") + enabled false + } + } + + // chain the upload tasks so they run sequentially in order to run + // the cliCleanUpTask after the final upload task is run + if (previousCliTask != null) { + previousCliTask.configure { finalizedBy cliTask } + } else { + bundleTask.configure { finalizedBy cliTask } + } + previousCliTask = cliTask + cliTask.configure { finalizedBy modulesTask } + } + + def modulesCleanUpTask = tasks.register(nameModulesCleanup, Delete) { + description = "clean up collected modules generated file" + group = 'sentry.io' + + delete modulesOutput + } + + /** Delete sourcemap files */ + def cliCleanUpTask = tasks.register(nameCleanup, Delete) { + description = "clean up extra sourcemap" + group = 'sentry.io' + + delete sourcemapOutput + delete "$buildDir/intermediates/assets/release/index.android.bundle.map" + // react native default bundle dir + } + + // register clean task extension + cliCleanUpTask.configure { onlyIf { shouldCleanUp } } + // due to chaining the last value of previousCliTask will be the final + // upload task, after which the cleanup can be done + previousCliTask.configure { finalizedBy cliCleanUpTask } + + def packageTasks = tasks.matching { + task -> ("package${applicationVariant}".equalsIgnoreCase(task.name) || "package${applicationVariant}Bundle".equalsIgnoreCase(task.name)) && task.enabled + } + packageTasks.configureEach { packageTask -> + packageTask.dependsOn modulesTask + packageTask.finalizedBy modulesCleanUpTask + } + } + } + } +} + // gradle.projectsEvaluated doesn't work with --configure-on-demand // the task are create too late and not executed project.afterEvaluate { @@ -95,257 +353,6 @@ project.afterEvaluate { println "* Flavor aware sentry properties *" println "**********************************" } - - // separately we then hook into the bundle task of react native to inject - // sourcemap generation parameters. In case for whatever reason no release - // was found for the asset folder we just bail. - def bundleTasks = tasks.findAll { task -> (task.name.startsWith("createBundle") || task.name.startsWith("bundle")) && task.name.endsWith("JsAndAssets") && !task.name.contains("Debug") && task.enabled } - bundleTasks.each { bundleTask -> - def shouldCleanUp - def sourcemapOutput - def bundleOutput - def packagerSourcemapOutput - def bundleCommand - def props = bundleTask.getProperties() - def reactRoot = props.get("workingDir") - if (reactRoot == null) { - reactRoot = props.get("root").get() // RN 0.71 and above - } - def modulesOutput = "$reactRoot/android/app/src/main/assets/modules.json" - def modulesTask = null - - (shouldCleanUp, bundleOutput, sourcemapOutput, packagerSourcemapOutput, bundleCommand) = forceSourceMapOutputFromBundleTask(bundleTask) - - // Lets leave this here if we need to debug - // println bundleTask.properties - // .sort{it.key} - // .collect{it} - // .findAll{!['class', 'active'].contains(it.key)} - // .join('\n') - - def currentVariants = extractCurrentVariants(bundleTask, releases) - if (currentVariants == null) return - - def previousCliTask = null - def applicationVariant = null - - def nameCleanup = "${bundleTask.name}_SentryUploadCleanUp" - def nameModulesCleanup = "${bundleTask.name}_SentryCollectModulesCleanUp" - // Upload the source map several times if necessary: once for each release and versionCode. - currentVariants.each { key, currentVariant -> - def variant = currentVariant[0] - def releaseName = currentVariant[1] - def versionCode = currentVariant[2] - applicationVariant = currentVariant[3] - - try { - if (versionCode instanceof String) { - versionCode = Integer.parseInt(versionCode) - versionCode = Math.abs(versionCode) - } - } catch (NumberFormatException e) { - project.logger.info("versionCode: '$versionCode' isn't an Integer, using the plain value.") - } - - // The Sentry server distinguishes source maps by release (`--release` in the command - // below) and distribution identifier (`--dist` below). Give the task a unique name - // based on where we're uploading to. - def nameCliTask = "${bundleTask.name}_SentryUpload_${releaseName}_${versionCode}" - def nameModulesTask = "${bundleTask.name}_SentryCollectModules_${releaseName}_${versionCode}" - - // If several outputs have the same releaseName and versionCode, we'd do the exact same - // upload for each of them. No need to repeat. - try { tasks.named(nameCliTask); return } catch (Exception e) {} - - /** Upload source map file to the sentry server via CLI call. */ - def cliTask = tasks.create(nameCliTask) { - onlyIf { shouldSentryAutoUploadGeneral() } - description = "upload debug symbols to sentry" - group = 'sentry.io' - - def extraArgs = [] - - def sentryPackage = resolveSentryReactNativeSDKPath(reactRoot) - def copyDebugIdScript = config.copyDebugIdScript - ? file(config.copyDebugIdScript).getAbsolutePath() - : "$sentryPackage/scripts/copy-debugid.js" - def hasSourceMapDebugIdScript = config.hasSourceMapDebugIdScript - ? file(config.hasSourceMapDebugIdScript).getAbsolutePath() - : "$sentryPackage/scripts/has-sourcemap-debugid.js" - - doFirst { - // Copy Debug ID from packager source map to Hermes composed source map - exec { - def args = ["node", - copyDebugIdScript, - packagerSourcemapOutput, - sourcemapOutput] - def osCompatibilityCopyCommand = Os.isFamily(Os.FAMILY_WINDOWS) ? ['cmd', '/c'] : [] - commandLine(*osCompatibilityCopyCommand, *args) - } - - // Add release and dist for backward compatibility if no Debug ID detected in output soruce map - def process = ["node", hasSourceMapDebugIdScript, sourcemapOutput].execute(null, new File("$reactRoot")) - def exitValue = process.waitFor() - project.logger.lifecycle("Check generated source map for Debug ID: ${process.text}") - - project.logger.lifecycle("Sentry Source Maps upload will include the release name and dist.") - extraArgs.addAll([ - "--release", releaseName, - "--dist", versionCode - ]) - } - - doLast { - exec { - workingDir reactRoot - - def propertiesFile = config.sentryProperties - ? config.sentryProperties - : "$reactRoot/android/sentry.properties" - - if (config.flavorAware) { - propertiesFile = "$reactRoot/android/sentry-${variant}.properties" - project.logger.info("For $variant using: $propertiesFile") - } else { - environment("SENTRY_PROPERTIES", propertiesFile) - } - - Properties sentryProps = new Properties() - try { - sentryProps.load(new FileInputStream(propertiesFile)) - } catch (FileNotFoundException e) { - project.logger.info("file not found '$propertiesFile' for '$variant'") - } - - def resolvedCliPackage = null - try { - resolvedCliPackage = new File(["node", "--print", "require.resolve('@sentry/cli/package.json')"].execute(null, rootDir).text.trim()).getParentFile(); - } catch (Throwable ignored) {} - def cliPackage = resolvedCliPackage != null && resolvedCliPackage.exists() ? resolvedCliPackage.getAbsolutePath() : "$reactRoot/node_modules/@sentry/cli" - def cliExecutable = sentryProps.get("cli.executable", "$cliPackage/bin/sentry-cli") - - // fix path separator for Windows - if (Os.isFamily(Os.FAMILY_WINDOWS)) { - cliExecutable = cliExecutable.replaceAll("/", "\\\\") - } - - // - // based on: - // https://github.com/getsentry/sentry-cli/blob/master/src/commands/react_native_gradle.rs - // - def args = [cliExecutable] - - args.addAll(!config.logLevel ? [] : [ - "--log-level", config.logLevel // control verbosity of the output - ]) - args.addAll(!config.flavorAware ? [] : [ - "--url", sentryProps.get("defaults.url"), - "--auth-token", sentryProps.get("auth.token") ?: System.getenv("SENTRY_AUTH_TOKEN") - ]) - args.addAll(["react-native", "gradle", - "--bundle", bundleOutput, // The path to a bundle that should be uploaded. - "--sourcemap", sourcemapOutput // The path to a sourcemap that should be uploaded. - ]) - args.addAll(!config.flavorAware ? [] : [ - "--org", sentryProps.get("defaults.org"), - "--project", sentryProps.get("defaults.project") - ]) - - args.addAll(extraArgs) - - project.logger.lifecycle("Sentry-CLI arguments: ${args}") - def osCompatibility = Os.isFamily(Os.FAMILY_WINDOWS) ? ['cmd', '/c', 'node'] : [] - if (!System.getenv('SENTRY_DOTENV_PATH')) { - environment('SENTRY_DOTENV_PATH', "$reactRoot/.env.sentry-build-plugin") - } - commandLine(*osCompatibility, *args) - } - } - - enabled true - } - - modulesTask = tasks.create(nameModulesTask, Exec) { - description = "collect javascript modules from bundle source map" - group = 'sentry.io' - - workingDir reactRoot - - def sentryPackage = resolveSentryReactNativeSDKPath(reactRoot) - - def collectModulesScript = config.collectModulesScript - ? file(config.collectModulesScript).getAbsolutePath() - : "$sentryPackage/dist/js/tools/collectModules.js" - def modulesPaths = config.modulesPaths - ? config.modulesPaths.join(',') - : "$reactRoot/node_modules" - def args = ["node", - collectModulesScript, - sourcemapOutput, - modulesOutput, - modulesPaths - ] - - if ((new File(collectModulesScript)).exists()) { - project.logger.info("Sentry-CollectModules arguments: ${args}") - commandLine(*args) - - def skip = config.skipCollectModules - ? config.skipCollectModules == true - : false - enabled !skip - } else { - project.logger.info("collectModulesScript not found: $collectModulesScript") - enabled false - } - } - - // chain the upload tasks so they run sequentially in order to run - // the cliCleanUpTask after the final upload task is run - if (previousCliTask != null) { - previousCliTask.finalizedBy cliTask - } else { - bundleTask.finalizedBy cliTask - } - previousCliTask = cliTask - cliTask.finalizedBy modulesTask - } - - def modulesCleanUpTask = tasks.create(name: nameModulesCleanup, type: Delete) { - description = "clean up collected modules generated file" - group = 'sentry.io' - - delete modulesOutput - } - - def packageTasks = tasks.findAll { - task -> ( - "package${applicationVariant}".equalsIgnoreCase(task.name) - || "package${applicationVariant}Bundle".equalsIgnoreCase(task.name) - ) && task.enabled - } - packageTasks.each { packageTask -> - packageTask.dependsOn modulesTask - packageTask.finalizedBy modulesCleanUpTask - } - - /** Delete sourcemap files */ - def cliCleanUpTask = tasks.create(name: nameCleanup, type: Delete) { - description = "clean up extra sourcemap" - group = 'sentry.io' - - delete sourcemapOutput - delete "$buildDir/intermediates/assets/release/index.android.bundle.map" - // react native default bundle dir - } - - // register clean task extension - cliCleanUpTask.onlyIf { shouldCleanUp } - // due to chaining the last value of previousCliTask will be the final - // upload task, after which the cleanup can be done - previousCliTask.finalizedBy cliCleanUpTask - } } def resolveSentryReactNativeSDKPath(reactRoot) { @@ -357,27 +364,29 @@ def resolveSentryReactNativeSDKPath(reactRoot) { return sentryPackage } -/** Compose lookup map of build variants - to - outputs. */ -def extractReleasesInfo() { - def releases = [:] - - android.applicationVariants.each { variant -> - - variant.outputs.each { output -> - def defaultVersionCode = output.getVersionCode() - def versionCode = System.getenv("SENTRY_DIST") ?: defaultVersionCode - def defaultReleaseName = "${variant.getApplicationId()}@${variant.getVersionName()}+${versionCode}" - def releaseName = System.getenv("SENTRY_RELEASE") ?: defaultReleaseName - def variantName = variant.getName() - def outputName = output.getName() - if (releases[variantName] == null) { - releases[variantName] = [:] +def resolveSentryCliPackagePath(reactRoot) { + def resolvedCliPath = null + try { + resolvedCliPath = new File(["node", "--print", "require.resolve('@sentry/cli/package.json')"].execute(null, rootDir).text.trim()).getParentFile(); + } catch (Throwable ignored) { // Check if it's located in .pnpm + try { + def pnpmRefPath = reactRoot.toString() + "/node_modules/@sentry/react-native/node_modules/.bin/sentry-cli" + def sentryCliFile = new File(pnpmRefPath) + + if (sentryCliFile.exists()) { + def cliFileText = sentryCliFile.text + def matcher = cliFileText =~ /NODE_PATH="([^"]*?)@sentry\/cli\// + + if (matcher.find()) { + def match = matcher.group(1) + resolvedCliPath = new File(match + "@sentry/cli") + } } - releases[variantName][outputName] = [outputName, releaseName, versionCode, variantName] - } + } catch (Throwable ignored2) {} // if the resolve fails we fallback to the default path } - return releases + def cliPackage = resolvedCliPath != null && resolvedCliPath.exists() ? resolvedCliPath.getAbsolutePath() : "$reactRoot/node_modules/@sentry/cli" + return cliPackage } /** Extract from arguments collection bundle and sourcemap files output names. */ @@ -491,7 +500,7 @@ def forceSourceMapOutputFromBundleTask(bundleTask) { } /** compose array with one item - current build flavor name */ -static extractCurrentVariants(bundleTask, releases) { +static extractCurrentVariants(bundleTask, variant) { // examples: bundleLocalReleaseJsAndAssets, createBundleYellowDebugJsAndAssets def pattern = Pattern.compile("(?:create)?(?:B|b)undle([A-Z][A-Za-z0-9_]+)JsAndAssets") @@ -504,9 +513,21 @@ static extractCurrentVariants(bundleTask, releases) { } def currentVariants = null - releases.each { key, release -> - if (key.equalsIgnoreCase(currentRelease)) { - currentVariants = release + if (variant.name.equalsIgnoreCase(currentRelease)) { + currentVariants = [:] + def variantName = variant.name + variant.outputs.each { output -> + def defaultVersionCode = output.versionCode.getOrElse(0) + def versionCode = System.getenv('SENTRY_DIST') ?: defaultVersionCode + def appId = variant.applicationId.get() + def versionName = output.versionName.getOrElse('') // may be empty if not set + def defaultReleaseName = "${appId}@${versionName}+${versionCode}" + def releaseName = System.getenv('SENTRY_RELEASE') ?: defaultReleaseName + + def outputName = output.baseName + + if (currentVariants[outputName] == null) currentVariants[outputName] = [] + currentVariants[outputName] = [outputName, releaseName, versionCode, variantName] } } diff --git a/packages/core/src/js/NativeRNSentry.ts b/packages/core/src/js/NativeRNSentry.ts index 5b00b62116..cdfbb0d781 100644 --- a/packages/core/src/js/NativeRNSentry.ts +++ b/packages/core/src/js/NativeRNSentry.ts @@ -1,7 +1,6 @@ import type { Package } from '@sentry/core'; import type { TurboModule } from 'react-native'; import { TurboModuleRegistry } from 'react-native'; - import type { UnsafeObject } from './utils/rnlibrariesinterface'; // There has to be only one interface and it has to be named `Spec` @@ -25,6 +24,7 @@ export interface Spec extends TurboModule { fetchNativeRelease(): Promise; fetchNativeSdkInfo(): Promise; fetchNativeDeviceContexts(): Promise; + fetchNativeLogAttributes(): Promise; fetchNativeAppStart(): Promise; fetchNativeFrames(): Promise; initNativeSdk(options: UnsafeObject): Promise; diff --git a/packages/core/src/js/client.ts b/packages/core/src/js/client.ts index 6fcad6e513..b263689735 100644 --- a/packages/core/src/js/client.ts +++ b/packages/core/src/js/client.ts @@ -10,9 +10,15 @@ import type { TransportMakeRequestResponse, UserFeedback, } from '@sentry/core'; -import { BaseClient, dateTimestampInSeconds, logger, SentryError } from '@sentry/core'; +import { + _INTERNAL_flushLogsBuffer, + addAutoIpAddressToSession, + Client, + dateTimestampInSeconds, + debug, + SentryError, +} from '@sentry/core'; import { Alert } from 'react-native'; - import { getDevServer } from './integrations/debugsymbolicatorutils'; import { defaultSdkInfo } from './integrations/sdkinfo'; import { getDefaultSidecarUrl } from './integrations/spotlight'; @@ -25,14 +31,17 @@ import { mergeOutcomes } from './utils/outcome'; import { ReactNativeLibraries } from './utils/rnlibraries'; import { NATIVE } from './wrapper'; +const DEFAULT_FLUSH_INTERVAL = 5000; + /** * The Sentry React Native SDK Client. * * @see ReactNativeClientOptions for documentation on configuration options. * @see SentryClient for usage documentation. */ -export class ReactNativeClient extends BaseClient { +export class ReactNativeClient extends Client { private _outcomesBuffer: Outcome[]; + private _logFlushIdleTimeout: ReturnType | undefined; /** * Creates a new React Native SDK instance. @@ -40,14 +49,44 @@ export class ReactNativeClient extends BaseClient { */ public constructor(options: ReactNativeClientOptions) { ignoreRequireCycleLogs(ReactNativeLibraries.ReactNativeVersion?.version); - options._metadata = options._metadata || {}; - options._metadata.sdk = options._metadata.sdk || defaultSdkInfo; + options._metadata = { + ...options._metadata, + sdk: { + ...(options._metadata?.sdk || defaultSdkInfo), + settings: { + // Only allow IP inferral by Relay if sendDefaultPii is true + infer_ip: options.sendDefaultPii ? 'auto' : 'never', + ...options._metadata?.sdk?.settings, + }, + }, + }; + // We default this to true, as it is the safer scenario options.parentSpanIsAlwaysRootSpan = options.parentSpanIsAlwaysRootSpan === undefined ? true : options.parentSpanIsAlwaysRootSpan; super(options); this._outcomesBuffer = []; + + if (options.sendDefaultPii === true) { + this.on('beforeSendSession', addAutoIpAddressToSession); + } + + if (options.enableLogs) { + this.on('flush', () => { + _INTERNAL_flushLogsBuffer(this); + }); + + this.on('afterCaptureLog', () => { + if (this._logFlushIdleTimeout) { + clearTimeout(this._logFlushIdleTimeout); + } + + this._logFlushIdleTimeout = setTimeout(() => { + _INTERNAL_flushLogsBuffer(this); + }, DEFAULT_FLUSH_INTERVAL); + }); + } } /** @@ -78,7 +117,7 @@ export class ReactNativeClient extends BaseClient { public close(): PromiseLike { // As super.close() flushes queued events, we wait for that to finish before closing the native SDK. return super.close().then((result: boolean) => { - return NATIVE.closeNativeSdk().then(() => result) as PromiseLike; + return NATIVE.closeNativeSdk().then(() => result); }); } @@ -116,13 +155,13 @@ export class ReactNativeClient extends BaseClient { // SentryError is thrown by SyncPromise shouldClearOutcomesBuffer = false; // If this is called asynchronously we want the _outcomesBuffer to be cleared - logger.error('SentryError while sending event, keeping outcomes buffer:', reason); + debug.error('SentryError while sending event, keeping outcomes buffer:', reason); } else { - logger.error('Error while sending event:', reason); + debug.error('Error while sending event:', reason); } }); } else { - logger.error('Transport disabled'); + debug.error('Transport disabled'); } if (shouldClearOutcomesBuffer) { @@ -188,7 +227,7 @@ export class ReactNativeClient extends BaseClient { this.emit('afterInit'); }) .then(undefined, error => { - logger.error('The OnReady callback threw an error: ', error); + debug.error('The OnReady callback threw an error: ', error); }); } diff --git a/packages/core/src/js/feedback/FeedbackButton.tsx b/packages/core/src/js/feedback/FeedbackButton.tsx index fbb546db8d..66ab86b07f 100644 --- a/packages/core/src/js/feedback/FeedbackButton.tsx +++ b/packages/core/src/js/feedback/FeedbackButton.tsx @@ -1,7 +1,6 @@ import * as React from 'react'; import type { NativeEventSubscription} from 'react-native'; import { Appearance, Image, Text, TouchableOpacity } from 'react-native'; - import { defaultButtonConfiguration } from './defaults'; import { defaultButtonStyles } from './FeedbackWidget.styles'; import { getTheme } from './FeedbackWidget.theme'; @@ -15,7 +14,7 @@ import { lazyLoadFeedbackIntegration } from './lazy'; * Implements a feedback button that opens the FeedbackForm. */ export class FeedbackButton extends React.Component { - private _themeListener: NativeEventSubscription; + private _themeListener: NativeEventSubscription | undefined; public constructor(props: FeedbackButtonProps) { super(props); @@ -58,8 +57,10 @@ export class FeedbackButton extends React.Component { onPress={showFeedbackWidget} accessibilityLabel={text.triggerAriaLabel} > - - {text.triggerLabel} + + + {text.triggerLabel} + ); } diff --git a/packages/core/src/js/feedback/FeedbackWidget.styles.ts b/packages/core/src/js/feedback/FeedbackWidget.styles.ts index 94df799d21..8620d8c9b3 100644 --- a/packages/core/src/js/feedback/FeedbackWidget.styles.ts +++ b/packages/core/src/js/feedback/FeedbackWidget.styles.ts @@ -1,5 +1,4 @@ import type { ViewStyle } from 'react-native'; - import type { FeedbackWidgetTheme } from './FeedbackWidget.theme'; import type { FeedbackButtonStyles, FeedbackWidgetStyles } from './FeedbackWidget.types'; diff --git a/packages/core/src/js/feedback/FeedbackWidget.theme.ts b/packages/core/src/js/feedback/FeedbackWidget.theme.ts index aa8711a934..602b6bdea3 100644 --- a/packages/core/src/js/feedback/FeedbackWidget.theme.ts +++ b/packages/core/src/js/feedback/FeedbackWidget.theme.ts @@ -1,5 +1,4 @@ import { Appearance } from 'react-native'; - import { getColorScheme, getFeedbackDarkTheme, getFeedbackLightTheme } from './integration'; /** diff --git a/packages/core/src/js/feedback/FeedbackWidget.tsx b/packages/core/src/js/feedback/FeedbackWidget.tsx index b84dd3b1b9..2725b3747a 100644 --- a/packages/core/src/js/feedback/FeedbackWidget.tsx +++ b/packages/core/src/js/feedback/FeedbackWidget.tsx @@ -1,6 +1,6 @@ /* eslint-disable max-lines */ -import type { SendFeedbackParams } from '@sentry/core'; -import { captureFeedback, getCurrentScope, lastEventId, logger } from '@sentry/core'; +import type { SendFeedbackParams, User } from '@sentry/core'; +import { captureFeedback, debug, getCurrentScope, getGlobalScope, getIsolationScope, lastEventId } from '@sentry/core'; import * as React from 'react'; import type { KeyboardTypeOptions , NativeEventSubscription} from 'react-native'; @@ -14,7 +14,6 @@ import { TouchableWithoutFeedback, View } from 'react-native'; - import { isExpoGo, isWeb, notWeb } from '../utils/environment'; import type { Screenshot } from '../wrapper'; import { getDataFromUri, NATIVE } from '../wrapper'; @@ -33,9 +32,7 @@ import { base64ToUint8Array, feedbackAlertDialog, isValidEmail } from './utils' * Implements a feedback form screen that sends feedback to Sentry using Sentry.captureFeedback. */ export class FeedbackWidget extends React.Component { - public static defaultProps: Partial = { - ...defaultConfiguration - } + public static defaultProps = defaultConfiguration; private static _savedState: Omit = { name: '', @@ -46,7 +43,7 @@ export class FeedbackWidget extends React.Component void = () => { const { name, email, description } = this.state; const { onSubmitSuccess, onSubmitError, onFormSubmitted } = this.props; - const text: FeedbackTextConfiguration = this.props; + const text = this.props; const trimmedName = name?.trim(); const trimmedEmail = email?.trim(); const trimmedDescription = description?.trim(); - if ((this.props.isNameRequired && !trimmedName) || (this.props.isEmailRequired && !trimmedEmail) || !trimmedDescription) { + if ( + (this.props.isNameRequired && !trimmedName) || + (this.props.isEmailRequired && !trimmedEmail) || + !trimmedDescription + ) { feedbackAlertDialog(text.errorTitle, text.formError); return; } - if (this.props.shouldValidateEmail && (this.props.isEmailRequired || trimmedEmail.length > 0) && !isValidEmail(trimmedEmail)) { + if ( + this.props.shouldValidateEmail && + (this.props.isEmailRequired || trimmedEmail.length > 0) && + !isValidEmail(trimmedEmail) + ) { feedbackAlertDialog(text.errorTitle, text.emailError); return; } - const attachments = this.state.filename && this.state.attachment - ? [ - { - filename: this.state.filename, - data: this.state.attachment, - }, - ] - : undefined; + const attachments = + this.state.filename && this.state.attachment + ? [ + { + filename: this.state.filename, + data: this.state.attachment, + }, + ] + : undefined; const eventId = lastEventId(); const userFeedback: SendFeedbackParams = { @@ -128,31 +134,36 @@ export class FeedbackWidget extends React.Component void = async () => { if (!this._hasScreenshot()) { - const imagePickerConfiguration: ImagePickerConfiguration = this.props; - if (imagePickerConfiguration.imagePicker) { - const launchImageLibrary = imagePickerConfiguration.imagePicker.launchImageLibraryAsync - // expo-image-picker library is available - ? () => imagePickerConfiguration.imagePicker.launchImageLibraryAsync({ mediaTypes: ['images'], base64: isWeb() }) - // react-native-image-picker library is available - : imagePickerConfiguration.imagePicker.launchImageLibrary - ? () => imagePickerConfiguration.imagePicker.launchImageLibrary({ mediaType: 'photo', includeBase64: isWeb() }) - : null; + const { imagePicker } = this.props; + if (imagePicker) { + const launchImageLibrary = imagePicker.launchImageLibraryAsync + ? // expo-image-picker library is available + () => imagePicker.launchImageLibraryAsync?.({ mediaTypes: ['images'], base64: isWeb() }) + : // react-native-image-picker library is available + imagePicker.launchImageLibrary + ? () => imagePicker.launchImageLibrary?.({ mediaType: 'photo', includeBase64: isWeb() }) + : null; if (!launchImageLibrary) { - logger.warn('No compatible image picker library found. Please provide a valid image picker library.'); + debug.warn('No compatible image picker library found. Please provide a valid image picker library.'); if (__DEV__) { feedbackAlertDialog( 'Development note', @@ -163,31 +174,34 @@ export class FeedbackWidget extends React.Component 0) { + if (result?.assets && result.assets.length > 0) { if (isWeb()) { - const filename = result.assets[0].fileName; - const imageUri = result.assets[0].uri; - const base64 = result.assets[0].base64; - const data = base64ToUint8Array(base64); - if (data != null) { + const filename = result.assets[0]?.fileName; + const imageUri = result.assets[0]?.uri; + const base64 = result.assets[0]?.base64; + const data = base64 ? base64ToUint8Array(base64) : undefined; + if (data) { this.setState({ filename, attachment: data, attachmentUri: imageUri }); } else { - logger.error('Failed to read image data on the web'); + debug.error('Failed to read image data on the web'); } } else { - const filename = result.assets[0].fileName; - const imageUri = result.assets[0].uri; - getDataFromUri(imageUri).then((data) => { - if (data != null) { - this.setState({ filename, attachment: data, attachmentUri: imageUri }); - } else { - this._showImageRetrievalDevelopmentNote(); - logger.error('Failed to read image data from uri:', imageUri); - } - }).catch((error) => { - this._showImageRetrievalDevelopmentNote(); - logger.error('Failed to read image data from uri:', imageUri, 'error: ', error); - }); + const filename = result.assets[0]?.fileName; + const imageUri = result.assets[0]?.uri; + imageUri && + getDataFromUri(imageUri) + .then((data) => { + if (data != null) { + this.setState({ filename, attachment: data, attachmentUri: imageUri }); + } else { + this._showImageRetrievalDevelopmentNote(); + debug.error('Failed to read image data from uri:', imageUri); + } + }) + .catch((error) => { + this._showImageRetrievalDevelopmentNote(); + debug.error('Failed to read image data from uri:', imageUri, 'error: ', error); + }); } } } else { @@ -199,18 +213,18 @@ export class FeedbackWidget extends React.Component { this._showImageRetrievalDevelopmentNote(); - logger.error('Failed to read image data from uri:', uri, 'error: ', error); + debug.error('Failed to read image data from uri:', uri, 'error: ', error); }); }); } } else { this.setState({ filename: undefined, attachment: undefined, attachmentUri: undefined }); } - } + }; /** * Add a listener to the theme change event. @@ -253,7 +267,7 @@ export class FeedbackWidget extends React.Component + > - {text.formTitle} + + {text.formTitle} + {config.showBranding && ( - + )} {config.showName && ( - <> - - {text.nameLabel} - {config.isNameRequired && ` ${text.isRequiredLabel}`} - - this.setState({ name: value })} - /> - + <> + + {text.nameLabel} + {config.isNameRequired && ` ${text.isRequiredLabel}`} + + this.setState({ name: value })} + /> + )} {config.showEmail && ( - <> - - {text.emailLabel} - {config.isEmailRequired && ` ${text.isRequiredLabel}`} - - this.setState({ email: value })} - /> - + <> + + {text.emailLabel} + {config.isEmailRequired && ` ${text.isRequiredLabel}`} + + this.setState({ email: value })} + /> + )} @@ -325,25 +337,20 @@ export class FeedbackWidget extends React.Component this.setState({ description: value })} + onChangeText={value => this.setState({ description: value })} multiline /> {(config.enableScreenshot || imagePickerConfiguration.imagePicker || this._hasScreenshot()) && ( {this.state.attachmentUri && ( - + )} - {!this._hasScreenshot() - ? text.addScreenshotButtonLabel - : text.removeScreenshotButtonLabel} + {!this._hasScreenshot() ? text.addScreenshotButtonLabel : text.removeScreenshotButtonLabel} @@ -354,11 +361,13 @@ export class FeedbackWidget extends React.Component - {text.captureScreenshotButtonLabel} + {text.captureScreenshotButtonLabel} )} - {text.submitButtonLabel} + + {text.submitButtonLabel} + @@ -371,21 +380,23 @@ export class FeedbackWidget extends React.Component { if (screenshot.data != null) { - logger.debug('Setting captured screenshot:', screenshot.filename); - NATIVE.encodeToBase64(screenshot.data).then((base64String) => { - if (base64String != null) { - const dataUri = `data:${screenshot.contentType};base64,${base64String}`; - this.setState({ filename: screenshot.filename, attachment: screenshot.data, attachmentUri: dataUri }); - } else { - logger.error('Failed to read image data from:', screenshot.filename); - } - }).catch((error) => { - logger.error('Failed to read image data from:', screenshot.filename, 'error: ', error); - }); + debug.log('Setting captured screenshot:', screenshot.filename); + NATIVE.encodeToBase64(screenshot.data) + .then(base64String => { + if (base64String != null) { + const dataUri = `data:${screenshot.contentType};base64,${base64String}`; + this.setState({ filename: screenshot.filename, attachment: screenshot.data, attachmentUri: dataUri }); + } else { + debug.error('Failed to read image data from:', screenshot.filename); + } + }) + .catch(error => { + debug.error('Failed to read image data from:', screenshot.filename, 'error: ', error); + }); } else { - logger.error('Failed to read image data from:', screenshot.filename); + debug.error('Failed to read image data from:', screenshot.filename); } - } + }; private _saveFormState = (): void => { FeedbackWidget._savedState = { ...this.state }; @@ -406,6 +417,18 @@ export class FeedbackWidget extends React.Component { + const currentUser = getCurrentScope().getUser(); + if (currentUser) { + return currentUser; + } + const isolationUser = getIsolationScope().getUser(); + if (isolationUser) { + return isolationUser; + } + return getGlobalScope().getUser(); + } + private _showImageRetrievalDevelopmentNote = (): void => { if (isExpoGo()) { feedbackAlertDialog( diff --git a/packages/core/src/js/feedback/FeedbackWidget.types.ts b/packages/core/src/js/feedback/FeedbackWidget.types.ts index 22b6b0911f..d3878dcfb0 100644 --- a/packages/core/src/js/feedback/FeedbackWidget.types.ts +++ b/packages/core/src/js/feedback/FeedbackWidget.types.ts @@ -21,38 +21,38 @@ export interface FeedbackGeneralConfiguration { * * @default true */ - showBranding?: boolean; + showBranding: boolean; /** * Should the email field be required? */ - isEmailRequired?: boolean; + isEmailRequired: boolean; /** * Should the email field be validated? */ - shouldValidateEmail?: boolean; + shouldValidateEmail: boolean; /** * Should the name field be required? */ - isNameRequired?: boolean; + isNameRequired: boolean; /** * Should the email input field be visible? Note: email will still be collected if set via `Sentry.setUser()` */ - showEmail?: boolean; + showEmail: boolean; /** * Should the name input field be visible? Note: name will still be collected if set via `Sentry.setUser()` */ - showName?: boolean; + showName: boolean; /** * This flag determines whether the "Add Screenshot" button is displayed * @default false */ - enableScreenshot?: boolean; + enableScreenshot: boolean; /** * This flag determines whether the "Take Screenshot" button is displayed @@ -77,32 +77,32 @@ export interface FeedbackTextConfiguration { /** * The label for the Feedback form cancel button that closes dialog */ - cancelButtonLabel?: string; + cancelButtonLabel: string; /** * The label for the Feedback form submit button that sends feedback */ - submitButtonLabel?: string; + submitButtonLabel: string; /** * The title of the Feedback form */ - formTitle?: string; + formTitle: string; /** * Label for the email input */ - emailLabel?: string; + emailLabel: string; /** * Placeholder text for Feedback email input */ - emailPlaceholder?: string; + emailPlaceholder: string; /** * Label for the message input */ - messageLabel?: string; + messageLabel: string; /** * Placeholder text for Feedback message input @@ -112,32 +112,32 @@ export interface FeedbackTextConfiguration { /** * Label for the name input */ - nameLabel?: string; + nameLabel: string; /** * Message after feedback was sent successfully */ - successMessageText?: string; + successMessageText: string; /** * Placeholder text for Feedback name input */ - namePlaceholder?: string; + namePlaceholder: string; /** * Text which indicates that a field is required */ - isRequiredLabel?: string; + isRequiredLabel: string; /** * The label for the button that adds a screenshot */ - addScreenshotButtonLabel?: string; + addScreenshotButtonLabel: string; /** * The label for the button that removes a screenshot */ - removeScreenshotButtonLabel?: string; + removeScreenshotButtonLabel: string; /** * The label for the button that shows the capture screenshot button @@ -147,27 +147,27 @@ export interface FeedbackTextConfiguration { /** * The title of the error dialog */ - errorTitle?: string; + errorTitle: string; /** * The error message when the form is invalid */ - formError?: string; + formError: string; /** * The error message when the email is invalid */ - emailError?: string; + emailError: string; /** * The error message when the capture screenshot fails */ - captureScreenshotError?: string; + captureScreenshotError: string; /** * Message when there is a generic error */ - genericError?: string; + genericError: string; } /** @@ -207,34 +207,34 @@ export interface FeedbackCallbacks { /** * Callback when form is opened */ - onFormOpen?: () => void; + onFormOpen: () => void; /** * Callback when form is closed and not submitted */ - onFormClose?: () => void; + onFormClose: () => void; /** * Callback when a screenshot is added */ - onAddScreenshot?: (addScreenshot: (uri: string) => void) => void; + onAddScreenshot: (addScreenshot: (uri: string) => void) => void; /** * Callback when feedback is successfully submitted * * After this you'll see a SuccessMessage on the screen for a moment. */ - onSubmitSuccess?: (data: FeedbackFormData) => void; + onSubmitSuccess: (data: FeedbackFormData) => void; /** * Callback when feedback is unsuccessfully submitted */ - onSubmitError?: (error: Error) => void; + onSubmitError: (error: Error) => void; /** * Callback when the feedback form is submitted successfully, and the SuccessMessage is complete, or dismissed */ - onFormSubmitted?: () => void; + onFormSubmitted: () => void; } /** diff --git a/packages/core/src/js/feedback/FeedbackWidgetManager.tsx b/packages/core/src/js/feedback/FeedbackWidgetManager.tsx index e554715586..505bf5e6da 100644 --- a/packages/core/src/js/feedback/FeedbackWidgetManager.tsx +++ b/packages/core/src/js/feedback/FeedbackWidgetManager.tsx @@ -1,5 +1,4 @@ -import { logger } from '@sentry/core'; - +import { debug } from '@sentry/core'; import { isWeb } from '../utils/environment'; import { lazyLoadAutoInjectFeedbackButtonIntegration,lazyLoadAutoInjectFeedbackIntegration, lazyLoadAutoInjectScreenshotButtonIntegration } from './lazy'; @@ -7,6 +6,10 @@ export const PULL_DOWN_CLOSE_THRESHOLD = 200; export const SLIDE_ANIMATION_DURATION = 200; export const BACKGROUND_ANIMATION_DURATION = 200; +const NOOP_SET_VISIBILITY = (): void => { + // No-op +}; + abstract class FeedbackManager { protected static _isVisible = false; protected static _setVisibility: (visible: boolean) => void; @@ -24,11 +27,11 @@ abstract class FeedbackManager { */ public static reset(): void { this._isVisible = false; - this._setVisibility = undefined; + this._setVisibility = NOOP_SET_VISIBILITY; } public static show(): void { - if (this._setVisibility) { + if (this._setVisibility !== NOOP_SET_VISIBILITY) { this._isVisible = true; this._setVisibility(true); } else { @@ -39,7 +42,7 @@ abstract class FeedbackManager { } public static hide(): void { - if (this._setVisibility) { + if (this._setVisibility !== NOOP_SET_VISIBILITY) { this._isVisible = false; this._setVisibility(false); } else { @@ -114,7 +117,7 @@ const resetFeedbackButtonManager = (): void => { const showScreenshotButton = (): void => { if (isWeb()) { - logger.warn('ScreenshotButton is not supported on Web.'); + debug.warn('ScreenshotButton is not supported on Web.'); return; } lazyLoadAutoInjectScreenshotButtonIntegration(); diff --git a/packages/core/src/js/feedback/FeedbackWidgetProvider.tsx b/packages/core/src/js/feedback/FeedbackWidgetProvider.tsx index 9e90ed785f..1c2c8bab73 100644 --- a/packages/core/src/js/feedback/FeedbackWidgetProvider.tsx +++ b/packages/core/src/js/feedback/FeedbackWidgetProvider.tsx @@ -1,14 +1,20 @@ -import { logger } from '@sentry/core'; +import { debug } from '@sentry/core'; import * as React from 'react'; import { type NativeEventSubscription, type NativeScrollEvent,type NativeSyntheticEvent, Animated, Appearance, Dimensions, Easing, Modal, PanResponder, Platform, ScrollView, View } from 'react-native'; - import { notWeb } from '../utils/environment'; import { FeedbackButton } from './FeedbackButton'; import { FeedbackWidget } from './FeedbackWidget'; import { modalSheetContainer,modalWrapper, topSpacer } from './FeedbackWidget.styles'; import { getTheme } from './FeedbackWidget.theme'; import type { FeedbackWidgetStyles } from './FeedbackWidget.types'; -import { BACKGROUND_ANIMATION_DURATION,FeedbackButtonManager, FeedbackWidgetManager, PULL_DOWN_CLOSE_THRESHOLD, ScreenshotButtonManager, SLIDE_ANIMATION_DURATION } from './FeedbackWidgetManager'; +import { + BACKGROUND_ANIMATION_DURATION, + FeedbackButtonManager, + FeedbackWidgetManager, + PULL_DOWN_CLOSE_THRESHOLD, + ScreenshotButtonManager, + SLIDE_ANIMATION_DURATION, +} from './FeedbackWidgetManager'; import { getFeedbackButtonOptions, getFeedbackOptions, getScreenshotButtonOptions } from './integration'; import { ScreenshotButton } from './ScreenshotButton'; import { isModalSupported, isNativeDriverSupportedForColorAnimations } from './utils'; @@ -44,7 +50,7 @@ export class FeedbackWidgetProvider extends React.Component { @@ -120,9 +126,9 @@ export class FeedbackWidgetProvider extends React.Component { - logger.info('FeedbackWidgetProvider componentDidUpdate'); + debug.log('FeedbackWidgetProvider componentDidUpdate'); }); } else if (prevState.isVisible && !this.state.isVisible) { this.state.backgroundOpacity.setValue(0); @@ -134,7 +140,7 @@ export class FeedbackWidgetProvider extends React.Component{this.props.children}; } @@ -154,25 +160,36 @@ export class FeedbackWidgetProvider extends React.Component} {isScreenshotButtonVisible && } - {isVisible && + {isVisible && ( - + + {...this._panResponder.panHandlers} + > - + + onFormSubmitted={this._handleClose} + /> - } + + )} ); } @@ -198,7 +215,7 @@ export class FeedbackWidgetProvider extends React.Component { // Change of the state unmount the component // which would cancel the animation diff --git a/packages/core/src/js/feedback/ScreenshotButton.tsx b/packages/core/src/js/feedback/ScreenshotButton.tsx index 18cfa19239..7f5bffd334 100644 --- a/packages/core/src/js/feedback/ScreenshotButton.tsx +++ b/packages/core/src/js/feedback/ScreenshotButton.tsx @@ -1,7 +1,6 @@ import * as React from 'react'; import type { NativeEventSubscription} from 'react-native'; import { Appearance, Image, Text, TouchableOpacity } from 'react-native'; - import type { Screenshot } from '../wrapper'; import { NATIVE } from '../wrapper'; import { defaultScreenshotButtonConfiguration } from './defaults'; @@ -38,7 +37,7 @@ export const getCapturedScreenshot = (): Screenshot | 'ErrorCapturingScreenshot' * Implements a screenshot button that takes a screenshot. */ export class ScreenshotButton extends React.Component { - private _themeListener: NativeEventSubscription; + private _themeListener: NativeEventSubscription | undefined; public constructor(props: ScreenshotButtonProps) { super(props); @@ -78,11 +77,11 @@ export class ScreenshotButton extends React.Component { return ( - {text.triggerLabel} + {text.triggerLabel} ); } diff --git a/packages/core/src/js/feedback/defaults.ts b/packages/core/src/js/feedback/defaults.ts index 2158b69a41..59f2092f9f 100644 --- a/packages/core/src/js/feedback/defaults.ts +++ b/packages/core/src/js/feedback/defaults.ts @@ -23,7 +23,7 @@ const CAPTURE_SCREENSHOT_LABEL = 'Take a screenshot'; const REMOVE_SCREENSHOT_LABEL = 'Remove screenshot'; const GENERIC_ERROR_TEXT = 'Unable to send feedback due to an unexpected error.'; -export const defaultConfiguration: Partial = { +export const defaultConfiguration: FeedbackWidgetProps = { // FeedbackCallbacks onFormOpen: () => { // Does nothing by default diff --git a/packages/core/src/js/feedback/integration.ts b/packages/core/src/js/feedback/integration.ts index d450422aa3..7182205278 100644 --- a/packages/core/src/js/feedback/integration.ts +++ b/packages/core/src/js/feedback/integration.ts @@ -1,5 +1,4 @@ import { type Integration, getClient } from '@sentry/core'; - import type { FeedbackWidgetTheme } from './FeedbackWidget.theme'; import type { FeedbackButtonProps, FeedbackWidgetProps, ScreenshotButtonProps } from './FeedbackWidget.types'; @@ -15,7 +14,7 @@ type FeedbackIntegration = Integration & { }; export const feedbackIntegration = ( - initOptions: FeedbackWidgetProps & { + initOptions: Partial & { buttonOptions?: FeedbackButtonProps; screenshotButtonOptions?: ScreenshotButtonProps; colorScheme?: 'system' | 'light' | 'dark'; @@ -43,7 +42,7 @@ export const feedbackIntegration = ( }; }; -const _getClientIntegration = (): FeedbackIntegration => { +const _getClientIntegration = (): FeedbackIntegration | undefined => { return getClient()?.getIntegrationByName>(MOBILE_FEEDBACK_INTEGRATION_NAME); }; @@ -76,7 +75,7 @@ export const getScreenshotButtonOptions = (): Partial => export const getColorScheme = (): 'system' | 'light' | 'dark' => { const integration = _getClientIntegration(); - if (!integration) { + if (!integration?.colorScheme) { return 'system'; } diff --git a/packages/core/src/js/feedback/lazy.ts b/packages/core/src/js/feedback/lazy.ts index c3d2b2727d..6bfad02f56 100644 --- a/packages/core/src/js/feedback/lazy.ts +++ b/packages/core/src/js/feedback/lazy.ts @@ -1,5 +1,4 @@ import { getClient } from '@sentry/core'; - import { feedbackIntegration, MOBILE_FEEDBACK_INTEGRATION_NAME } from './integration'; /** diff --git a/packages/core/src/js/feedback/utils.ts b/packages/core/src/js/feedback/utils.ts index 6644bd7468..be839957ae 100644 --- a/packages/core/src/js/feedback/utils.ts +++ b/packages/core/src/js/feedback/utils.ts @@ -1,5 +1,4 @@ import { Alert } from 'react-native'; - import { isFabricEnabled, isWeb } from '../utils/environment'; import { RN_GLOBAL_OBJ } from '../utils/worldwide'; import { ReactNativeLibraries } from './../utils/rnlibraries'; @@ -15,7 +14,7 @@ declare global { */ export function isModalSupported(): boolean { const { major, minor } = ReactNativeLibraries.ReactNativeVersion?.version || {}; - return !(isFabricEnabled() && major === 0 && minor < 71); + return !(isFabricEnabled() && major === 0 && minor && minor < 71); } /** @@ -24,7 +23,7 @@ export function isModalSupported(): boolean { */ export function isNativeDriverSupportedForColorAnimations(): boolean { const { major, minor } = ReactNativeLibraries.ReactNativeVersion?.version || {}; - return major > 0 || minor >= 69; + return (major && major > 0) || (minor && minor >= 69) || false; } export const isValidEmail = (email: string): boolean => { diff --git a/packages/core/src/js/index.ts b/packages/core/src/js/index.ts index 1a163a2b06..4a475a33c1 100644 --- a/packages/core/src/js/index.ts +++ b/packages/core/src/js/index.ts @@ -1,6 +1,5 @@ export type { Breadcrumb, - Request, SdkInfo, Event, Exception, @@ -18,6 +17,7 @@ export type { export { addBreadcrumb, + addIntegration, captureException, captureEvent, captureFeedback, @@ -45,7 +45,6 @@ export { getClient, setCurrentClient, addEventProcessor, - metricsDefault as metrics, lastEventId, } from '@sentry/core'; @@ -58,13 +57,20 @@ export { withProfiler, } from '@sentry/react'; +export { + logger, + consoleLoggingIntegration, + featureFlagsIntegration, + type FeatureFlagsIntegration, +} from '@sentry/browser'; + export * from './integrations/exports'; export { SDK_NAME, SDK_VERSION } from './version'; export type { ReactNativeOptions } from './options'; export { ReactNativeClient } from './client'; -export { init, wrap, nativeCrash, flush, close, captureUserFeedback, withScope, crashedLastRun } from './sdk'; +export { init, wrap, nativeCrash, flush, close, withScope, crashedLastRun } from './sdk'; export { TouchEventBoundary, withTouchEventBoundary } from './touchevents'; export { diff --git a/packages/core/src/js/integrations/appRegistry.ts b/packages/core/src/js/integrations/appRegistry.ts index 2467d73876..73041354c0 100644 --- a/packages/core/src/js/integrations/appRegistry.ts +++ b/packages/core/src/js/integrations/appRegistry.ts @@ -1,6 +1,5 @@ import type { Client, Integration } from '@sentry/core'; -import { getClient, logger } from '@sentry/core'; - +import { debug, getClient } from '@sentry/core'; import { isWeb } from '../utils/environment'; import { fillTyped } from '../utils/fill'; import { ReactNativeLibraries } from '../utils/rnlibraries'; @@ -23,7 +22,7 @@ export const appRegistryIntegration = (): Integration & { }, onRunApplication: (callback: () => void) => { if (callbacks.includes(callback)) { - logger.debug('[AppRegistryIntegration] Callback already registered.'); + debug.log('[AppRegistryIntegration] Callback already registered.'); return; } callbacks.push(callback); diff --git a/packages/core/src/js/integrations/breadcrumbs.ts b/packages/core/src/js/integrations/breadcrumbs.ts new file mode 100644 index 0000000000..da3b57eceb --- /dev/null +++ b/packages/core/src/js/integrations/breadcrumbs.ts @@ -0,0 +1,74 @@ +import { breadcrumbsIntegration as browserBreadcrumbsIntegration } from '@sentry/browser'; +import type { Integration } from '@sentry/core'; +import { isWeb } from '../utils/environment'; + +interface BreadcrumbsOptions { + /** + * Log calls to console.log, console.debug, and so on. + */ + console: boolean; + + /** + * Log all click and keypress events. + * + * Only available on web. In React Native this is a no-op. + */ + dom: + | boolean + | { + serializeAttribute?: string | string[]; + maxStringLength?: number; + }; + + /** + * Log HTTP requests done with the global Fetch API. + * + * Disabled by default in React Native because fetch is built on XMLHttpRequest. + * Enabled by default on web. + * + * Setting `fetch: true` and `xhr: true` will cause duplicates in React Native. + */ + fetch: boolean; + + /** + * Log calls to history.pushState and related APIs. + * + * Only available on web. In React Native this is a no-op. + */ + history: boolean; + + /** + * Log whenever we send an event to the server. + */ + sentry: boolean; + + /** + * Log HTTP requests done with the XHR API. + * + * Because React Native global fetch is built on XMLHttpRequest, + * this will also log `fetch` network requests. + * + * Setting `fetch: true` and `xhr: true` will cause duplicates in React Native. + */ + xhr: boolean; +} + +export const breadcrumbsIntegration = (options: Partial = {}): Integration => { + const _options: BreadcrumbsOptions = { + // FIXME: In mobile environment XHR is implemented by native APIs, which are instrumented by the Native SDK. + // This will cause duplicates in React Native. On iOS `NSURLSession` is instrumented by default. On Android + // `OkHttp` is only instrumented by SAGP. + xhr: true, + console: true, + sentry: true, + ...options, + fetch: options.fetch ?? (isWeb() ? true : false), + dom: isWeb() ? options.dom ?? true : false, + history: isWeb() ? options.history ?? true : false, + }; + + // Historically we had very little issue using the browser breadcrumbs integration + // and thus we don't cherry pick the implementation like for example the Sentry Deno SDK does. + // https://github.com/getsentry/sentry-javascript/blob/d007407c2e51d93d6d3933f9dea1e03ff3f4a4ab/packages/deno/src/integrations/breadcrumbs.ts#L34 + return browserBreadcrumbsIntegration(_options); +}; diff --git a/packages/core/src/js/integrations/debugsymbolicator.ts b/packages/core/src/js/integrations/debugsymbolicator.ts index 8529d0eeb6..8daafd0005 100644 --- a/packages/core/src/js/integrations/debugsymbolicator.ts +++ b/packages/core/src/js/integrations/debugsymbolicator.ts @@ -1,6 +1,5 @@ import type { Event, EventHint, Exception, Integration, StackFrame as SentryStackFrame } from '@sentry/core'; -import { logger } from '@sentry/core'; - +import { debug } from '@sentry/core'; import type { ExtendedError } from '../utils/error'; import { getFramesToPop, isErrorLike } from '../utils/error'; import type * as ReactNative from '../vendor/react-native'; @@ -71,7 +70,7 @@ async function symbolicate(rawStack: string, skipFirstFrames: number = 0): Promi const prettyStack = await symbolicateStackTrace(parsedStack); if (!prettyStack) { - logger.error('React Native DevServer could not symbolicate the stack trace.'); + debug.error('React Native DevServer could not symbolicate the stack trace.'); return null; } @@ -93,7 +92,7 @@ async function symbolicate(rawStack: string, skipFirstFrames: number = 0): Promi return await fetchSourceContext(sentryFrames); } catch (error) { if (error instanceof Error) { - logger.warn(`Unable to symbolicate stack trace: ${error.message}`); + debug.warn(`Unable to symbolicate stack trace: ${error.message}`); } return null; } @@ -131,7 +130,7 @@ async function convertReactNativeFramesToSentryFrames(frames: ReactNative.StackF * @param event Event * @param frames StackFrame[] */ -function replaceExceptionFramesInException(exception: Exception, frames: SentryStackFrame[]): void { +function replaceExceptionFramesInException(exception: Exception | undefined, frames: SentryStackFrame[]): void { if (exception?.stacktrace) { exception.stacktrace.frames = frames.reverse(); } @@ -143,7 +142,7 @@ function replaceExceptionFramesInException(exception: Exception, frames: SentryS * @param frames StackFrame[] */ function replaceThreadFramesInEvent(event: Event, frames: SentryStackFrame[]): void { - if (event.threads && event.threads.values && event.threads.values[0] && event.threads.values[0].stacktrace) { + if (event.threads?.values?.[0]?.stacktrace) { event.threads.values[0].stacktrace.frames = frames.reverse(); } } diff --git a/packages/core/src/js/integrations/debugsymbolicatorutils.ts b/packages/core/src/js/integrations/debugsymbolicatorutils.ts index 2b51171b39..18c595efca 100644 --- a/packages/core/src/js/integrations/debugsymbolicatorutils.ts +++ b/packages/core/src/js/integrations/debugsymbolicatorutils.ts @@ -1,6 +1,5 @@ import type { StackFrame as SentryStackFrame } from '@sentry/core'; -import { logger } from '@sentry/core'; - +import { debug } from '@sentry/core'; import { ReactNativeLibraries } from '../utils/rnlibraries'; import { createStealthXhr, XHR_READYSTATE_DONE } from '../utils/xhr'; import type * as ReactNative from '../vendor/react-native'; @@ -20,7 +19,14 @@ export async function fetchSourceContext(frames: SentryStackFrame[]): Promise { // Ensures native errors and crashes have the same context as JS errors NATIVE.setContext(OTA_UPDATES_CONTEXT_KEY, expoUpdates); } catch (error) { - logger.error('Error setting Expo updates context:', error); + debug.error('Error setting Expo updates context:', error); } } diff --git a/packages/core/src/js/integrations/exports.ts b/packages/core/src/js/integrations/exports.ts index e87a88c615..4f4d0fb0ac 100644 --- a/packages/core/src/js/integrations/exports.ts +++ b/packages/core/src/js/integrations/exports.ts @@ -22,9 +22,11 @@ export { userInteractionIntegration } from '../tracing/integrations/userInteract export { createReactNativeRewriteFrames } from './rewriteframes'; export { appRegistryIntegration } from './appRegistry'; export { timeToDisplayIntegration } from '../tracing/integrations/timeToDisplayIntegration'; +export { breadcrumbsIntegration } from './breadcrumbs'; +export { primitiveTagIntegration } from './primitiveTagIntegration'; +export { logEnricherIntegration } from './logEnricherIntegration'; export { - breadcrumbsIntegration, browserApiErrorsIntegration, dedupeIntegration, functionToStringIntegration, diff --git a/packages/core/src/js/integrations/logEnricherIntegration.ts b/packages/core/src/js/integrations/logEnricherIntegration.ts new file mode 100644 index 0000000000..285365cc77 --- /dev/null +++ b/packages/core/src/js/integrations/logEnricherIntegration.ts @@ -0,0 +1,95 @@ +/* eslint-disable complexity */ +import type { Integration, Log } from '@sentry/core'; +import { debug } from '@sentry/core'; +import type { ReactNativeClient } from '../client'; +import { NATIVE } from '../wrapper'; + +const INTEGRATION_NAME = 'LogEnricher'; + +export const logEnricherIntegration = (): Integration => { + return { + name: INTEGRATION_NAME, + setup(client: ReactNativeClient) { + client.on('afterInit', () => { + cacheLogContext().then( + () => { + client.on('beforeCaptureLog', (log: Log) => { + processLog(log, client); + }); + }, + reason => { + debug.log(reason); + }, + ); + }); + }, + }; +}; + +let NativeCache: Record | undefined = undefined; + +/** + * Sets a log attribute if the value exists and the attribute key is not already present. + * + * @param logAttributes - The log attributes object to modify. + * @param key - The attribute key to set. + * @param value - The value to set (only sets if truthy and key not present). + * @param setEvenIfPresent - Whether to set the attribute if it is present. Defaults to true. + */ +function setLogAttribute( + logAttributes: Record, + key: string, + value: unknown, + setEvenIfPresent = true, +): void { + if (value && (!logAttributes[key] || setEvenIfPresent)) { + logAttributes[key] = value; + } +} + +async function cacheLogContext(): Promise { + try { + const response = await NATIVE.fetchNativeLogAttributes(); + + NativeCache = { + ...(response?.contexts?.device && { + brand: response.contexts.device?.brand, + model: response.contexts.device?.model, + family: response.contexts.device?.family, + }), + ...(response?.contexts?.os && { + os: response.contexts.os.name, + version: response.contexts.os.version, + }), + ...(response?.contexts?.release && { + release: response.contexts.release, + }), + }; + } catch (e) { + return Promise.reject(`[LOGS]: Failed to prepare attributes from Native Layer: ${e}`); + } + return Promise.resolve(); +} + +function processLog(log: Log, client: ReactNativeClient): void { + if (NativeCache === undefined) { + return; + } + + // Save log.attributes to a new variable + const logAttributes = log.attributes ?? {}; + + // Use setLogAttribute with the variable instead of direct assignment + setLogAttribute(logAttributes, 'device.brand', NativeCache.brand); + setLogAttribute(logAttributes, 'device.model', NativeCache.model); + setLogAttribute(logAttributes, 'device.family', NativeCache.family); + setLogAttribute(logAttributes, 'os.name', NativeCache.os); + setLogAttribute(logAttributes, 'os.version', NativeCache.version); + setLogAttribute(logAttributes, 'sentry.release', NativeCache.release); + + const replay = client.getIntegrationByName string | null }>('MobileReplay'); + setLogAttribute(logAttributes, 'sentry.replay_id', replay?.getReplayId()); + + // Set log.attributes to the variable + log.attributes = logAttributes; +} diff --git a/packages/core/src/js/integrations/modulesloader.ts b/packages/core/src/js/integrations/modulesloader.ts index 7a31154d33..d08ec6ebc9 100644 --- a/packages/core/src/js/integrations/modulesloader.ts +++ b/packages/core/src/js/integrations/modulesloader.ts @@ -1,6 +1,5 @@ import type { Event, Integration } from '@sentry/core'; -import { logger } from '@sentry/core'; - +import { debug } from '@sentry/core'; import { NATIVE } from '../wrapper'; const INTEGRATION_NAME = 'ModulesLoader'; @@ -25,7 +24,7 @@ function createProcessEvent(): (event: Event) => Promise { try { modules = await NATIVE.fetchModules(); } catch (e) { - logger.log(`Failed to get modules from native: ${e}`); + debug.log(`Failed to get modules from native: ${e}`); } isSetup = true; } diff --git a/packages/core/src/js/integrations/nativelinkederrors.ts b/packages/core/src/js/integrations/nativelinkederrors.ts index 39d8d55879..727ef85638 100644 --- a/packages/core/src/js/integrations/nativelinkederrors.ts +++ b/packages/core/src/js/integrations/nativelinkederrors.ts @@ -11,7 +11,6 @@ import type { StackParser, } from '@sentry/core'; import { isInstanceOf, isPlainObject, isString } from '@sentry/core'; - import type { NativeStackFrames } from '../NativeRNSentry'; import { NATIVE } from '../wrapper'; @@ -43,7 +42,7 @@ export const nativeLinkedErrorsIntegration = (options: Partial True, false -> False. + * Symbols are stringified. + * + */ +export const primitiveTagIntegration = (): Integration => { + return { + name: INTEGRATION_NAME, + setup(client) { + client.on('beforeSendEvent', event => { + if (event.tags) { + Object.keys(event.tags).forEach(key => { + event.tags![key] = PrimitiveToString(event.tags![key]); + }); + } + }); + }, + afterAllSetup() { + if (NATIVE.enableNative) { + NATIVE._setPrimitiveProcessor((value: Primitive) => PrimitiveToString(value)); + } + }, + }; +}; diff --git a/packages/core/src/js/integrations/reactnativeerrorhandlers.ts b/packages/core/src/js/integrations/reactnativeerrorhandlers.ts index df56da47cb..f838717603 100644 --- a/packages/core/src/js/integrations/reactnativeerrorhandlers.ts +++ b/packages/core/src/js/integrations/reactnativeerrorhandlers.ts @@ -3,11 +3,11 @@ import { addExceptionMechanism, addGlobalUnhandledRejectionInstrumentationHandler, captureException, + debug, getClient, getCurrentScope, - logger, } from '@sentry/core'; - +import type { ReactNativeClientOptions } from '../options'; import { isHermesEnabled, isWeb } from '../utils/environment'; import { createSyntheticError, isErrorLike } from '../utils/error'; import { RN_GLOBAL_OBJ } from '../utils/worldwide'; @@ -58,7 +58,7 @@ function setupUnhandledRejectionsTracking(patchGlobalPromise: boolean): void { RN_GLOBAL_OBJ.HermesInternal?.enablePromiseRejectionTracker && RN_GLOBAL_OBJ?.HermesInternal?.hasPromise?.() ) { - logger.log('Using Hermes native promise rejection tracking'); + debug.log('Using Hermes native promise rejection tracking'); RN_GLOBAL_OBJ.HermesInternal.enablePromiseRejectionTracker({ allRejections: true, @@ -66,9 +66,9 @@ function setupUnhandledRejectionsTracking(patchGlobalPromise: boolean): void { onHandled: promiseRejectionTrackingOptions.onHandled, }); - logger.log('Unhandled promise rejections will be caught by Sentry.'); + debug.log('Unhandled promise rejections will be caught by Sentry.'); } else if (isWeb()) { - logger.log('Using Browser JS promise rejection tracking for React Native Web'); + debug.log('Using Browser JS promise rejection tracking for React Native Web'); // Use Sentry's built-in global unhandled rejection handler addGlobalUnhandledRejectionInstrumentationHandler((error: unknown) => { @@ -85,10 +85,10 @@ function setupUnhandledRejectionsTracking(patchGlobalPromise: boolean): void { checkPromiseAndWarn(); } else { // For JSC and other environments, patching was disabled by user configuration - logger.log('Unhandled promise rejections will not be caught by Sentry.'); + debug.log('Unhandled promise rejections will not be caught by Sentry.'); } } catch (e) { - logger.warn( + debug.warn( 'Failed to set up promise rejection tracking. ' + 'Unhandled promise rejections will not be caught by Sentry.' + 'See https://docs.sentry.io/platforms/react-native/troubleshooting/ for more details.', @@ -99,7 +99,7 @@ function setupUnhandledRejectionsTracking(patchGlobalPromise: boolean): void { const promiseRejectionTrackingOptions: PromiseRejectionTrackingOptions = { onUnhandled: (id, error: unknown, rejection = {}) => { if (__DEV__) { - logger.warn(`Possible Unhandled Promise Rejection (id: ${id}):\n${rejection}`); + debug.warn(`Possible Unhandled Promise Rejection (id: ${id}):\n${rejection}`); } // Marking the rejection as handled to avoid breaking crash rate calculations. @@ -113,7 +113,7 @@ const promiseRejectionTrackingOptions: PromiseRejectionTrackingOptions = { }, onHandled: id => { if (__DEV__) { - logger.warn( + debug.warn( `Promise Rejection Handled (id: ${id})\n` + 'This means you can ignore any previous messages of the form ' + `"Possible Unhandled Promise Rejection (id: ${id}):"`, @@ -137,11 +137,11 @@ function setupErrorUtilsGlobalHandler(): void { const errorUtils = RN_GLOBAL_OBJ.ErrorUtils; if (!errorUtils) { - logger.warn('ErrorUtils not found. Can be caused by different environment for example react-native-web.'); + debug.warn('ErrorUtils not found. Can be caused by different environment for example react-native-web.'); return; } - const defaultHandler = errorUtils.getGlobalHandler && errorUtils.getGlobalHandler(); + const defaultHandler = errorUtils.getGlobalHandler?.(); // eslint-disable-next-line @typescript-eslint/no-explicit-any errorUtils.setGlobalHandler(async (error: any, isFatal?: boolean) => { @@ -149,7 +149,7 @@ function setupErrorUtilsGlobalHandler(): void { const shouldHandleFatal = isFatal && !__DEV__; if (shouldHandleFatal) { if (handlingFatal) { - logger.log('Encountered multiple fatals in a row. The latest:', error); + debug.log('Encountered multiple fatals in a row. The latest:', error); return; } handlingFatal = true; @@ -158,7 +158,7 @@ function setupErrorUtilsGlobalHandler(): void { const client = getClient(); if (!client) { - logger.error('Sentry client is missing, the error event might be lost.', error); + debug.error('Sentry client is missing, the error event might be lost.', error); // If there is no client something is fishy, anyway we call the default handler defaultHandler(error, isFatal); @@ -197,12 +197,12 @@ function setupErrorUtilsGlobalHandler(): void { return; } - void client.flush(client.getOptions().shutdownTimeout || 2000).then( + void client.flush((client.getOptions() as ReactNativeClientOptions).shutdownTimeout || 2000).then( () => { defaultHandler(error, isFatal); }, (reason: unknown) => { - logger.error('[ReactNativeErrorHandlers] Error while flushing the event cache after uncaught error.', reason); + debug.error('[ReactNativeErrorHandlers] Error while flushing the event cache after uncaught error.', reason); }, ); }); diff --git a/packages/core/src/js/integrations/reactnativeerrorhandlersutils.ts b/packages/core/src/js/integrations/reactnativeerrorhandlersutils.ts index 7453f696d6..9b2bb4790f 100644 --- a/packages/core/src/js/integrations/reactnativeerrorhandlersutils.ts +++ b/packages/core/src/js/integrations/reactnativeerrorhandlersutils.ts @@ -1,5 +1,4 @@ -import { logger } from '@sentry/core'; - +import { debug } from '@sentry/core'; import { ReactNativeLibraries } from '../utils/rnlibraries'; import { RN_GLOBAL_OBJ } from '../utils/worldwide'; @@ -14,7 +13,7 @@ import { RN_GLOBAL_OBJ } from '../utils/worldwide'; */ export function polyfillPromise(): void { if (!ReactNativeLibraries.Utilities) { - logger.warn('Could not polyfill Promise. React Native Libraries Utilities not found.'); + debug.warn('Could not polyfill Promise. React Native Libraries Utilities not found.'); return; } @@ -66,7 +65,7 @@ export function checkPromiseAndWarn(): void { const UsedPromisePolyfill = getPromisePolyfill(); if (ReactNativePromise !== PromisePackagePromise) { - logger.warn( + debug.warn( 'You appear to have multiple versions of the "promise" package installed. ' + 'This may cause unexpected behavior like undefined `Promise.allSettled`. ' + 'Please install the `promise` package manually using the exact version as the React Native package. ' + @@ -76,16 +75,16 @@ export function checkPromiseAndWarn(): void { // This only make sense if the user disabled the integration Polyfill if (UsedPromisePolyfill !== RN_GLOBAL_OBJ.Promise) { - logger.warn( + debug.warn( 'Unhandled promise rejections will not be caught by Sentry. ' + 'See https://docs.sentry.io/platforms/react-native/troubleshooting/ for more details.', ); } else { - logger.log('Unhandled promise rejections will be caught by Sentry.'); + debug.log('Unhandled promise rejections will be caught by Sentry.'); } } catch (e) { // Do Nothing - logger.warn( + debug.warn( 'Unhandled promise rejections will not be caught by Sentry. ' + 'See https://docs.sentry.io/platforms/react-native/troubleshooting/ for more details.', ); diff --git a/packages/core/src/js/integrations/reactnativeinfo.ts b/packages/core/src/js/integrations/reactnativeinfo.ts index b24e20f917..f45139f232 100644 --- a/packages/core/src/js/integrations/reactnativeinfo.ts +++ b/packages/core/src/js/integrations/reactnativeinfo.ts @@ -1,5 +1,4 @@ import type { Context, Event, EventHint, Integration } from '@sentry/core'; - import { getExpoGoVersion, getExpoSdkVersion, @@ -61,7 +60,7 @@ function processEvent(event: Event, hint: EventHint): Event { if (reactNativeContext.js_engine === 'hermes') { event.tags = { - hermes: 'true', + hermes: true, ...event.tags, }; } diff --git a/packages/core/src/js/integrations/release.ts b/packages/core/src/js/integrations/release.ts index f414f8a9ac..3682bb42b3 100644 --- a/packages/core/src/js/integrations/release.ts +++ b/packages/core/src/js/integrations/release.ts @@ -1,5 +1,4 @@ import type { BaseTransportOptions, Client, ClientOptions, Event, EventHint, Integration } from '@sentry/core'; - import { NATIVE } from '../wrapper'; const INTEGRATION_NAME = 'Release'; diff --git a/packages/core/src/js/integrations/rewriteframes.ts b/packages/core/src/js/integrations/rewriteframes.ts index 09ee8b6398..81c28a3bf9 100644 --- a/packages/core/src/js/integrations/rewriteframes.ts +++ b/packages/core/src/js/integrations/rewriteframes.ts @@ -1,7 +1,6 @@ import type { Integration, StackFrame } from '@sentry/core'; import { rewriteFramesIntegration } from '@sentry/core'; import { Platform } from 'react-native'; - import { isExpo, isHermesEnabled } from '../utils/environment'; export const ANDROID_DEFAULT_BUNDLE_NAME = 'app:///index.android.bundle'; diff --git a/packages/core/src/js/integrations/screenshot.ts b/packages/core/src/js/integrations/screenshot.ts index 6f504fa76e..3c45ada451 100644 --- a/packages/core/src/js/integrations/screenshot.ts +++ b/packages/core/src/js/integrations/screenshot.ts @@ -1,5 +1,4 @@ import type { Event, EventHint, Integration } from '@sentry/core'; - import type { ReactNativeClient } from '../client'; import type { Screenshot as ScreenshotAttachment } from '../wrapper'; import { NATIVE } from '../wrapper'; @@ -18,7 +17,7 @@ export const screenshotIntegration = (): Integration => { }; async function processEvent(event: Event, hint: EventHint, client: ReactNativeClient): Promise { - const hasException = event.exception && event.exception.values && event.exception.values.length > 0; + const hasException = event.exception?.values && event.exception.values.length > 0; if (!hasException || client.getOptions().beforeScreenshot?.(event, hint) === false) { return event; } diff --git a/packages/core/src/js/integrations/sdkinfo.ts b/packages/core/src/js/integrations/sdkinfo.ts index b90614d5c3..f8d54a136c 100644 --- a/packages/core/src/js/integrations/sdkinfo.ts +++ b/packages/core/src/js/integrations/sdkinfo.ts @@ -1,6 +1,5 @@ import type { Event, Integration, Package, SdkInfo as SdkInfoType } from '@sentry/core'; -import { logger } from '@sentry/core'; - +import { debug } from '@sentry/core'; import { isExpoGo, notWeb } from '../utils/environment'; import { SDK_NAME, SDK_PACKAGE_NAME, SDK_VERSION } from '../version'; import { NATIVE } from '../wrapper'; @@ -41,7 +40,7 @@ async function processEvent(event: Event, fetchNativeSdkInfo: () => Promise Promise { nativeSdkPackageCache = await NATIVE.fetchNativeSdkInfo(); isCached = true; } catch (e) { - logger.warn('Could not fetch native sdk info.', e); + debug.warn('Could not fetch native sdk info.', e); } return nativeSdkPackageCache; diff --git a/packages/core/src/js/integrations/spotlight.ts b/packages/core/src/js/integrations/spotlight.ts index b4f62e06da..01f35bb819 100644 --- a/packages/core/src/js/integrations/spotlight.ts +++ b/packages/core/src/js/integrations/spotlight.ts @@ -1,6 +1,5 @@ import type { BaseTransportOptions, Client, ClientOptions, Envelope, Integration } from '@sentry/core'; -import { logger, serializeEnvelope } from '@sentry/core'; - +import { debug, serializeEnvelope } from '@sentry/core'; import { ReactNativeLibraries } from '../utils/rnlibraries'; import { createStealthXhr, XHR_READYSTATE_DONE } from '../utils/xhr'; @@ -22,7 +21,7 @@ type SpotlightReactNativeIntegrationOptions = { export function spotlightIntegration({ sidecarUrl = getDefaultSidecarUrl(), }: SpotlightReactNativeIntegrationOptions = {}): Integration { - logger.info('[Spotlight] Using Sidecar URL', sidecarUrl); + debug.log('[Spotlight] Using Sidecar URL', sidecarUrl); return { name: 'Spotlight', @@ -57,7 +56,7 @@ function sendEnvelopesToSidecar(client: Client, sidecarUrl: string): void { const xhr = createStealthXhr(); if (!xhr) { - logger.error('[Spotlight] Sentry SDK can not create XHR object'); + debug.error('[Spotlight] Sentry SDK can not create XHR object'); return; } @@ -71,7 +70,7 @@ function sendEnvelopesToSidecar(client: Client, sidecarUrl: string): void { // The request has been completed successfully } else { // Handle the error - logger.error( + debug.error( "[Spotlight] Sentry SDK can't connect to Spotlight is it running? See https://spotlightjs.com to download it.", new Error(xhr.statusText), ); @@ -83,17 +82,23 @@ function sendEnvelopesToSidecar(client: Client, sidecarUrl: string): void { }); } +const DEFAULT_SIDECAR_URL = 'http://localhost:8969/stream'; + /** * Gets the default Spotlight sidecar URL. */ export function getDefaultSidecarUrl(): string { try { - const { url } = ReactNativeLibraries.Devtools?.getDevServer(); + const { url } = ReactNativeLibraries.Devtools?.getDevServer() ?? {}; + if (!url) { + return DEFAULT_SIDECAR_URL; + } + return `http://${getHostnameFromString(url)}:8969/stream`; } catch (_oO) { // We can't load devserver URL } - return 'http://localhost:8969/stream'; + return DEFAULT_SIDECAR_URL; } /** @@ -103,7 +108,7 @@ function getHostnameFromString(urlString: string): string | null { const regex = /^(?:\w+:)?\/\/([^/:]+)(:\d+)?(.*)$/; const matches = urlString.match(regex); - if (matches && matches[1]) { + if (matches?.[1]) { return matches[1]; } else { // Invalid URL format diff --git a/packages/core/src/js/integrations/viewhierarchy.ts b/packages/core/src/js/integrations/viewhierarchy.ts index 9b84ece273..f05e76705f 100644 --- a/packages/core/src/js/integrations/viewhierarchy.ts +++ b/packages/core/src/js/integrations/viewhierarchy.ts @@ -1,6 +1,5 @@ import type { Attachment, Event, EventHint, Integration } from '@sentry/core'; -import { logger } from '@sentry/core'; - +import { debug } from '@sentry/core'; import { NATIVE } from '../wrapper'; const filename: string = 'view-hierarchy.json'; @@ -21,7 +20,7 @@ export const viewHierarchyIntegration = (): Integration => { }; async function processEvent(event: Event, hint: EventHint): Promise { - const hasException = event.exception && event.exception.values && event.exception.values.length > 0; + const hasException = event.exception?.values && event.exception.values.length > 0; if (!hasException) { return event; } @@ -30,7 +29,7 @@ async function processEvent(event: Event, hint: EventHint): Promise { try { viewHierarchy = await NATIVE.fetchViewHierarchy(); } catch (e) { - logger.error('Failed to get view hierarchy from native.', e); + debug.error('Failed to get view hierarchy from native.', e); } if (viewHierarchy) { diff --git a/packages/core/src/js/options.ts b/packages/core/src/js/options.ts index a95de6df4c..44bfecfd46 100644 --- a/packages/core/src/js/options.ts +++ b/packages/core/src/js/options.ts @@ -1,15 +1,17 @@ import type { makeFetchTransport } from '@sentry/browser'; import type { CaptureContext, ClientOptions, Event, EventHint, Options } from '@sentry/core'; -import type { Profiler } from '@sentry/react'; +import type { BrowserOptions, Profiler } from '@sentry/react'; import type * as React from 'react'; import { Platform } from 'react-native'; - import type { TouchEventBoundaryProps } from './touchevents'; -import { getExpoConstants } from './utils/expomodules'; +import { isExpoGo } from './utils/environment'; type ProfilerProps = React.ComponentProps; type BrowserTransportOptions = Parameters[0]; +type BrowserExperiments = NonNullable; +type SharedExperimentsSubset = BrowserExperiments; + export interface BaseReactNativeOptions { /** * Enables native transport + device info + offline caching. @@ -234,10 +236,26 @@ export interface BaseReactNativeOptions { */ replaysOnErrorSampleRate?: number; + /** + * Controls how many milliseconds to wait before shutting down. The default is 2 seconds. Setting this too low can cause + * problems for sending events from command line applications. Setting it too + * high can cause the application to block for users with network connectivity + * problems. + */ + shutdownTimeout?: number; + + /** + * Defines the quality of the session replay. The higher the quality, the more accurate the replay + * will be, but also more data to transfer and more CPU load. + * + * @default 'medium' + */ + replaysSessionQuality?: SentryReplayQuality; + /** * Options which are in beta, or otherwise not guaranteed to be stable. */ - _experiments?: { + _experiments?: SharedExperimentsSubset & { [key: string]: unknown; /** @@ -253,6 +271,17 @@ export interface BaseReactNativeOptions { * This will be removed in the next major version. */ replaysOnErrorSampleRate?: number; + + /** + * Experiment: A more reliable way to report unhandled C++ exceptions in iOS. + * + * This approach hooks into all instances of the `__cxa_throw` function, which provides a more comprehensive and consistent exception handling across an app’s runtime, regardless of the number of C++ modules or how they’re linked. It helps in obtaining accurate stack traces. + * + * - Note: The mechanism of hooking into `__cxa_throw` could cause issues with symbolication on iOS due to caching of symbol references. + * + * @default false + */ + enableUnhandledCPPExceptionsV2?: boolean; }; /** @@ -262,8 +291,24 @@ export interface BaseReactNativeOptions { * @deprecated This option will be removed in the next major version. Use `beforeSend` instead. */ useThreadsForMessageStack?: boolean; + + /** + * If set to `true`, the SDK propagates the W3C `traceparent` header to any outgoing requests, + * in addition to the `sentry-trace` and `baggage` headers. Use the {@link CoreOptions.tracePropagationTargets} + * option to control to which outgoing requests the header will be attached. + * + * **Important:** If you set this option to `true`, make sure that you configured your servers' + * CORS settings to allow the `traceparent` header. Otherwise, requests might get blocked. + * + * @see https://www.w3.org/TR/trace-context/ + * + * @default false + */ + propagateTraceparent?: boolean; } +export type SentryReplayQuality = 'low' | 'medium' | 'high'; + export interface ReactNativeTransportOptions extends BrowserTransportOptions { /** * @deprecated use `maxQueueSize` in the root of the SDK options. @@ -286,7 +331,7 @@ export interface ReactNativeClientOptions export interface ReactNativeWrapperOptions { /** Props for the root React profiler */ - profilerProps?: ProfilerProps; + profilerProps?: Omit; /** Props for the root touch event boundary */ touchEventBoundaryProps?: TouchEventBoundaryProps; @@ -308,8 +353,7 @@ export function shouldEnableNativeNagger(userOptions: unknown): boolean { return false; } - const expoConstants = getExpoConstants(); - if (expoConstants && expoConstants.appOwnership === 'expo') { + if (isExpoGo()) { // If the app is running in Expo Go, we don't want to nag return false; } diff --git a/packages/core/src/js/playground/animations.tsx b/packages/core/src/js/playground/animations.tsx new file mode 100644 index 0000000000..34d223c5fd --- /dev/null +++ b/packages/core/src/js/playground/animations.tsx @@ -0,0 +1,8 @@ +export const hi = + ''; + +export const bug = + ''; + +export const thumbsup = + ''; diff --git a/packages/core/src/js/playground/examples.ts b/packages/core/src/js/playground/examples.ts new file mode 100644 index 0000000000..f82ddca83c --- /dev/null +++ b/packages/core/src/js/playground/examples.ts @@ -0,0 +1,37 @@ +import { captureException } from '@sentry/core'; +import { NATIVE } from '../wrapper'; + +// This is a placeholder to match the example code with what Sentry SDK users would see. +const Sentry = { + captureException, + nativeCrash: (): void => { + NATIVE.nativeCrash(); + }, +}; + +/** + * Example of error handling with Sentry integration. + */ +export const tryCatchExample = (): void => { + try { + // If you see the line below highlighted the source maps are working correctly. + throw new Error('This is a test caught error.'); + } catch (e) { + Sentry.captureException(e); + } +}; + +/** + * Example of an uncaught error causing a crash from JS. + */ +export const uncaughtErrorExample = (): void => { + // If you see the line below highlighted the source maps are working correctly. + throw new Error('This is a test uncaught error.'); +}; + +/** + * Example of a native crash. + */ +export const nativeCrashExample = (): void => { + Sentry.nativeCrash(); +}; diff --git a/packages/core/src/js/playground/images.tsx b/packages/core/src/js/playground/images.tsx new file mode 100644 index 0000000000..66b2a61296 --- /dev/null +++ b/packages/core/src/js/playground/images.tsx @@ -0,0 +1,8 @@ +export const hi = + ''; + +export const bug = + ''; + +export const thumbsup = + ''; diff --git a/packages/core/src/js/playground/index.ts b/packages/core/src/js/playground/index.ts new file mode 100644 index 0000000000..2f27f62c7a --- /dev/null +++ b/packages/core/src/js/playground/index.ts @@ -0,0 +1 @@ +export { withSentryPlayground } from './modal'; diff --git a/packages/core/src/js/playground/modal.tsx b/packages/core/src/js/playground/modal.tsx new file mode 100644 index 0000000000..910f6b46b7 --- /dev/null +++ b/packages/core/src/js/playground/modal.tsx @@ -0,0 +1,437 @@ +/* eslint-disable max-lines */ +import { debug } from '@sentry/core'; +import * as React from 'react'; +import { + Animated, + Image, + Modal, + Platform, + Pressable, + StyleSheet, + Text, + useColorScheme, + View, +} from 'react-native'; +import { getDevServer } from '../integrations/debugsymbolicatorutils'; +import { isExpo, isExpoGo, isWeb } from '../utils/environment'; +import { bug as bugAnimation, hi as hiAnimation, thumbsup as thumbsupAnimation } from './animations'; +import { nativeCrashExample, tryCatchExample, uncaughtErrorExample } from './examples'; +import { bug as bugImage, hi as hiImage, thumbsup as thumbsupImage } from './images'; + +/** + * Wrapper to add Sentry Playground to your application + * to test your Sentry React Native SDK setup. + * + * @example + * ```tsx + * import * as Sentry from '@sentry/react-native'; + * import { withSentryPlayground } from '@sentry/react-native/playground'; + * + * function App() { + * return ...; + * } + * + * export default withSentryPlayground(Sentry.wrap(App), { + * projectId: '123456', + * organizationSlug: 'my-org' + * }); + * ``` + */ +export const withSentryPlayground =

( + Component: React.ComponentType

, + options: { projectId?: string; organizationSlug?: string } = {}, +): React.ComponentType

=> { + const Wrapper = (props: P): React.ReactElement => { + return ( + <> + + + + ); + }; + + Wrapper.displayName = 'withSentryPlayground()'; + return Wrapper; +}; + +export const SentryPlayground = ({ + projectId, + organizationSlug, +}: { + projectId?: string; + organizationSlug?: string; +}): React.ReactElement => { + const issuesStreamUrl = + projectId && organizationSlug + ? `https://${organizationSlug}.sentry.io/issues/?project=${projectId}&statsPeriod=1h` + : 'https://sentry.io/'; + const styles = useColorScheme() === 'dark' ? defaultDarkStyles : lightStyles; + + const [show, setShow] = React.useState(true); + const [animation, setAnimation] = React.useState('hi'); + + const onAnimationPress = (): void => { + switch (animation) { + case 'hi': + setAnimation('thumbsup'); + break; + default: + setAnimation('hi'); + } + }; + + const showOpenSentryButton = !isExpo(); + const isNativeCrashDisabled = isWeb() || isExpoGo() || __DEV__; + + const animationContainerYPosition = React.useRef(new Animated.Value(0)).current; + + const springAnimation = Animated.sequence([ + Animated.timing(animationContainerYPosition, { + toValue: -50, + duration: 300, + useNativeDriver: true, + }), + Animated.spring(animationContainerYPosition, { + toValue: 0, + friction: 4, + tension: 40, + useNativeDriver: true, + }), + ]); + + const changeAnimationToBug = (func: () => void): void => { + setAnimation('bug'); + springAnimation.start(() => { + func(); + }); + }; + + return ( + { + setShow(false); + }} + > + + + Welcome to Sentry Playground! + { + springAnimation.start(); + }} + > + + + + + + + changeAnimationToBug(uncaughtErrorExample)} + /> + + + + + {showOpenSentryButton && ( +